Add service for adding products to a Picnic order (#67877)
* Add Picnic services for searching products and adding products to the cart * Improve the Picnic services implementation and add unit tests * Fix pre-commit check issues * Fix comments and example product name * Remove search service, update add_product service schema * Fix pylint suggestion * Add more tests and removed unused code * Remove code needed for the removed service, clean tests from obvious comments and add type hints * Remove unused import * Remove unnecessary comments and simplify getting the config entry id Co-authored-by: Allen Porter <allen.porter@gmail.com> * Don't use hass.data in tests, make device id mandatory for service * Rewrite all service tests so using lru.cache is not needed * Add test for uncovered line in _product_search() * Require a config entry id as service parameter instead of device id * Use explicit check in get_api_client() and raise HomeAssistantError * Fix HomeAssistantError import, fix services tests * Change HomeAssistantError to ValueError when config entry is not found Co-authored-by: Allen Porter <allen.porter@gmail.com>
This commit is contained in:
parent
7ddf2e4ca4
commit
a848dc1155
5 changed files with 350 additions and 0 deletions
|
@ -8,6 +8,7 @@ from homeassistant.core import HomeAssistant
|
||||||
|
|
||||||
from .const import CONF_API, CONF_COORDINATOR, CONF_COUNTRY_CODE, DOMAIN
|
from .const import CONF_API, CONF_COORDINATOR, CONF_COUNTRY_CODE, DOMAIN
|
||||||
from .coordinator import PicnicUpdateCoordinator
|
from .coordinator import PicnicUpdateCoordinator
|
||||||
|
from .services import async_register_services
|
||||||
|
|
||||||
PLATFORMS = [Platform.SENSOR]
|
PLATFORMS = [Platform.SENSOR]
|
||||||
|
|
||||||
|
@ -36,6 +37,9 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
||||||
|
|
||||||
await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
|
await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
|
||||||
|
|
||||||
|
# Register the services
|
||||||
|
await async_register_services(hass)
|
||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -17,6 +17,14 @@ CONF_API = "api"
|
||||||
CONF_COORDINATOR = "coordinator"
|
CONF_COORDINATOR = "coordinator"
|
||||||
CONF_COUNTRY_CODE = "country_code"
|
CONF_COUNTRY_CODE = "country_code"
|
||||||
|
|
||||||
|
SERVICE_ADD_PRODUCT_TO_CART = "add_product"
|
||||||
|
|
||||||
|
ATTR_CONFIG_ENTRY_ID = "config_entry_id"
|
||||||
|
ATTR_PRODUCT_ID = "product_id"
|
||||||
|
ATTR_PRODUCT_NAME = "product_name"
|
||||||
|
ATTR_AMOUNT = "amount"
|
||||||
|
ATTR_PRODUCT_IDENTIFIERS = "product_identifiers"
|
||||||
|
|
||||||
COUNTRY_CODES = ["NL", "DE", "BE"]
|
COUNTRY_CODES = ["NL", "DE", "BE"]
|
||||||
ATTRIBUTION = "Data provided by Picnic"
|
ATTRIBUTION = "Data provided by Picnic"
|
||||||
ADDRESS = "address"
|
ADDRESS = "address"
|
||||||
|
|
90
homeassistant/components/picnic/services.py
Normal file
90
homeassistant/components/picnic/services.py
Normal file
|
@ -0,0 +1,90 @@
|
||||||
|
"""Services for the Picnic integration."""
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from python_picnic_api import PicnicAPI
|
||||||
|
import voluptuous as vol
|
||||||
|
|
||||||
|
from homeassistant.core import HomeAssistant, ServiceCall
|
||||||
|
import homeassistant.helpers.config_validation as cv
|
||||||
|
|
||||||
|
from .const import (
|
||||||
|
ATTR_AMOUNT,
|
||||||
|
ATTR_CONFIG_ENTRY_ID,
|
||||||
|
ATTR_PRODUCT_ID,
|
||||||
|
ATTR_PRODUCT_IDENTIFIERS,
|
||||||
|
ATTR_PRODUCT_NAME,
|
||||||
|
CONF_API,
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_ADD_PRODUCT_TO_CART,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class PicnicServiceException(Exception):
|
||||||
|
"""Exception for Picnic services."""
|
||||||
|
|
||||||
|
|
||||||
|
async def async_register_services(hass: HomeAssistant) -> None:
|
||||||
|
"""Register services for the Picnic integration, if not registered yet."""
|
||||||
|
|
||||||
|
if hass.services.has_service(DOMAIN, SERVICE_ADD_PRODUCT_TO_CART):
|
||||||
|
return
|
||||||
|
|
||||||
|
async def async_add_product_service(call: ServiceCall):
|
||||||
|
api_client = await get_api_client(hass, call.data[ATTR_CONFIG_ENTRY_ID])
|
||||||
|
await handle_add_product(hass, api_client, call)
|
||||||
|
|
||||||
|
hass.services.async_register(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_ADD_PRODUCT_TO_CART,
|
||||||
|
async_add_product_service,
|
||||||
|
schema=vol.Schema(
|
||||||
|
{
|
||||||
|
vol.Required(ATTR_CONFIG_ENTRY_ID): cv.string,
|
||||||
|
vol.Exclusive(
|
||||||
|
ATTR_PRODUCT_ID, ATTR_PRODUCT_IDENTIFIERS
|
||||||
|
): cv.positive_int,
|
||||||
|
vol.Exclusive(ATTR_PRODUCT_NAME, ATTR_PRODUCT_IDENTIFIERS): cv.string,
|
||||||
|
vol.Optional(ATTR_AMOUNT): vol.All(vol.Coerce(int), vol.Range(min=1)),
|
||||||
|
}
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def get_api_client(hass: HomeAssistant, config_entry_id: str) -> PicnicAPI:
|
||||||
|
"""Get the right Picnic API client based on the device id, else get the default one."""
|
||||||
|
if config_entry_id not in hass.data[DOMAIN]:
|
||||||
|
raise ValueError(f"Config entry with id {config_entry_id} not found!")
|
||||||
|
return hass.data[DOMAIN][config_entry_id][CONF_API]
|
||||||
|
|
||||||
|
|
||||||
|
async def handle_add_product(
|
||||||
|
hass: HomeAssistant, api_client: PicnicAPI, call: ServiceCall
|
||||||
|
) -> None:
|
||||||
|
"""Handle the call for the add_product service."""
|
||||||
|
product_id = call.data.get("product_id")
|
||||||
|
if not product_id:
|
||||||
|
product_id = await hass.async_add_executor_job(
|
||||||
|
_product_search, api_client, call.data.get("product_name")
|
||||||
|
)
|
||||||
|
|
||||||
|
if not product_id:
|
||||||
|
raise PicnicServiceException("No product found or no product ID given!")
|
||||||
|
|
||||||
|
await hass.async_add_executor_job(
|
||||||
|
api_client.add_product, str(product_id), call.data.get("amount", 1)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _product_search(api_client: PicnicAPI, product_name: str) -> None | str:
|
||||||
|
"""Query the api client for the product name."""
|
||||||
|
search_result = api_client.search(product_name)
|
||||||
|
|
||||||
|
if not search_result or "items" not in search_result[0]:
|
||||||
|
return None
|
||||||
|
|
||||||
|
# Return the first valid result
|
||||||
|
for item in search_result[0]["items"]:
|
||||||
|
if "name" in item:
|
||||||
|
return str(item["id"])
|
||||||
|
|
||||||
|
return None
|
37
homeassistant/components/picnic/services.yaml
Normal file
37
homeassistant/components/picnic/services.yaml
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
add_product:
|
||||||
|
name: Add a product to the cart
|
||||||
|
description: >-
|
||||||
|
Adds a product to the cart based on a search string or product ID.
|
||||||
|
The search string and product ID are exclusive.
|
||||||
|
|
||||||
|
fields:
|
||||||
|
config_entry_id:
|
||||||
|
name: Picnic service
|
||||||
|
description: The product will be added to the selected service.
|
||||||
|
required: true
|
||||||
|
selector:
|
||||||
|
config_entry:
|
||||||
|
integration: picnic
|
||||||
|
product_id:
|
||||||
|
name: Product ID
|
||||||
|
description: The product ID of a Picnic product.
|
||||||
|
required: false
|
||||||
|
example: "10510201"
|
||||||
|
selector:
|
||||||
|
text:
|
||||||
|
product_name:
|
||||||
|
name: Product name
|
||||||
|
description: Search for a product and add the first result
|
||||||
|
required: false
|
||||||
|
example: "Yoghurt"
|
||||||
|
selector:
|
||||||
|
text:
|
||||||
|
amount:
|
||||||
|
name: Amount
|
||||||
|
description: Amount to add of the selected product
|
||||||
|
required: false
|
||||||
|
default: 1
|
||||||
|
selector:
|
||||||
|
number:
|
||||||
|
min: 1
|
||||||
|
max: 50
|
211
tests/components/picnic/test_services.py
Normal file
211
tests/components/picnic/test_services.py
Normal file
|
@ -0,0 +1,211 @@
|
||||||
|
"""Tests for the Picnic services."""
|
||||||
|
from unittest.mock import MagicMock, patch
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from homeassistant.components.picnic import CONF_COUNTRY_CODE, DOMAIN
|
||||||
|
from homeassistant.components.picnic.const import SERVICE_ADD_PRODUCT_TO_CART
|
||||||
|
from homeassistant.components.picnic.services import PicnicServiceException
|
||||||
|
from homeassistant.const import CONF_ACCESS_TOKEN
|
||||||
|
from homeassistant.core import HomeAssistant
|
||||||
|
|
||||||
|
from tests.common import MockConfigEntry
|
||||||
|
|
||||||
|
UNIQUE_ID = "295-6y3-1nf4"
|
||||||
|
|
||||||
|
|
||||||
|
def create_picnic_api_client(unique_id):
|
||||||
|
"""Create PicnicAPI mock with set response data."""
|
||||||
|
auth_token = "af3wh738j3fa28l9fa23lhiufahu7l"
|
||||||
|
auth_data = {
|
||||||
|
"user_id": unique_id,
|
||||||
|
"address": {
|
||||||
|
"street": "Teststreet",
|
||||||
|
"house_number": 123,
|
||||||
|
"house_number_ext": "b",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
picnic_mock = MagicMock()
|
||||||
|
picnic_mock.session.auth_token = auth_token
|
||||||
|
picnic_mock.get_user.return_value = auth_data
|
||||||
|
|
||||||
|
return picnic_mock
|
||||||
|
|
||||||
|
|
||||||
|
async def create_picnic_config_entry(hass: HomeAssistant, unique_id):
|
||||||
|
"""Create a Picnic config entry."""
|
||||||
|
config_entry = MockConfigEntry(
|
||||||
|
domain=DOMAIN,
|
||||||
|
data={
|
||||||
|
CONF_ACCESS_TOKEN: "x-original-picnic-auth-token",
|
||||||
|
CONF_COUNTRY_CODE: "NL",
|
||||||
|
},
|
||||||
|
unique_id=unique_id,
|
||||||
|
)
|
||||||
|
config_entry.add_to_hass(hass)
|
||||||
|
await hass.config_entries.async_setup(config_entry.entry_id)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
return config_entry
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def picnic_api_client():
|
||||||
|
"""Return the default picnic api client."""
|
||||||
|
with patch(
|
||||||
|
"homeassistant.components.picnic.create_picnic_client"
|
||||||
|
) as create_picnic_client_mock:
|
||||||
|
picnic_client_mock = create_picnic_api_client(UNIQUE_ID)
|
||||||
|
create_picnic_client_mock.return_value = picnic_client_mock
|
||||||
|
|
||||||
|
yield picnic_client_mock
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
async def picnic_config_entry(hass: HomeAssistant):
|
||||||
|
"""Generate the default Picnic config entry."""
|
||||||
|
return await create_picnic_config_entry(hass, UNIQUE_ID)
|
||||||
|
|
||||||
|
|
||||||
|
async def test_add_product_using_id(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
picnic_api_client: MagicMock,
|
||||||
|
picnic_config_entry: MockConfigEntry,
|
||||||
|
):
|
||||||
|
"""Test adding a product by id."""
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_ADD_PRODUCT_TO_CART,
|
||||||
|
{
|
||||||
|
"config_entry_id": picnic_config_entry.entry_id,
|
||||||
|
"product_id": "5109348572",
|
||||||
|
"amount": 3,
|
||||||
|
},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Check that the right method is called on the api
|
||||||
|
picnic_api_client.add_product.assert_called_with("5109348572", 3)
|
||||||
|
|
||||||
|
|
||||||
|
async def test_add_product_using_name(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
picnic_api_client: MagicMock,
|
||||||
|
picnic_config_entry: MockConfigEntry,
|
||||||
|
):
|
||||||
|
"""Test adding a product by name."""
|
||||||
|
|
||||||
|
# Set the return value of the search api endpoint
|
||||||
|
picnic_api_client.search.return_value = [
|
||||||
|
{
|
||||||
|
"items": [
|
||||||
|
{
|
||||||
|
"id": "2525404",
|
||||||
|
"name": "Best tea",
|
||||||
|
"display_price": 321,
|
||||||
|
"unit_quantity": "big bags",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "2525500",
|
||||||
|
"name": "Cheap tea",
|
||||||
|
"display_price": 100,
|
||||||
|
"unit_quantity": "small bags",
|
||||||
|
},
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_ADD_PRODUCT_TO_CART,
|
||||||
|
{"config_entry_id": picnic_config_entry.entry_id, "product_name": "Tea"},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Check that the right method is called on the api
|
||||||
|
picnic_api_client.add_product.assert_called_with("2525404", 1)
|
||||||
|
|
||||||
|
|
||||||
|
async def test_add_product_using_name_no_results(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
picnic_api_client: MagicMock,
|
||||||
|
picnic_config_entry: MockConfigEntry,
|
||||||
|
):
|
||||||
|
"""Test adding a product by name that can't be found."""
|
||||||
|
|
||||||
|
# Set the search return value and check that the right exception is raised during the service call
|
||||||
|
picnic_api_client.search.return_value = []
|
||||||
|
with pytest.raises(PicnicServiceException):
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_ADD_PRODUCT_TO_CART,
|
||||||
|
{
|
||||||
|
"config_entry_id": picnic_config_entry.entry_id,
|
||||||
|
"product_name": "Random non existing product",
|
||||||
|
},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def test_add_product_using_name_no_named_results(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
picnic_api_client: MagicMock,
|
||||||
|
picnic_config_entry: MockConfigEntry,
|
||||||
|
):
|
||||||
|
"""Test adding a product by name for which no named results are returned."""
|
||||||
|
|
||||||
|
# Set the search return value and check that the right exception is raised during the service call
|
||||||
|
picnic_api_client.search.return_value = [{"items": [{"attr": "test"}]}]
|
||||||
|
with pytest.raises(PicnicServiceException):
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_ADD_PRODUCT_TO_CART,
|
||||||
|
{
|
||||||
|
"config_entry_id": picnic_config_entry.entry_id,
|
||||||
|
"product_name": "Random product",
|
||||||
|
},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def test_add_product_multiple_config_entries(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
picnic_api_client: MagicMock,
|
||||||
|
picnic_config_entry: MockConfigEntry,
|
||||||
|
):
|
||||||
|
"""Test adding a product for a specific Picnic service while multiple are configured."""
|
||||||
|
with patch(
|
||||||
|
"homeassistant.components.picnic.create_picnic_client"
|
||||||
|
) as create_picnic_client_mock:
|
||||||
|
picnic_api_client_2 = create_picnic_api_client("3fj9-9gju-236")
|
||||||
|
create_picnic_client_mock.return_value = picnic_api_client_2
|
||||||
|
picnic_config_entry_2 = await create_picnic_config_entry(hass, "3fj9-9gju-236")
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_ADD_PRODUCT_TO_CART,
|
||||||
|
{"product_id": "5109348572", "config_entry_id": picnic_config_entry_2.entry_id},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Check that the right method is called on the api
|
||||||
|
picnic_api_client.add_product.assert_not_called()
|
||||||
|
picnic_api_client_2.add_product.assert_called_with("5109348572", 1)
|
||||||
|
|
||||||
|
|
||||||
|
async def test_add_product_device_doesnt_exist(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
picnic_api_client: MagicMock,
|
||||||
|
picnic_config_entry: MockConfigEntry,
|
||||||
|
):
|
||||||
|
"""Test adding a product for a specific Picnic service, which doesn't exist."""
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_ADD_PRODUCT_TO_CART,
|
||||||
|
{"product_id": "5109348572", "config_entry_id": 12345},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Check that the right method is called on the api
|
||||||
|
picnic_api_client.add_product.assert_not_called()
|
Loading…
Add table
Add a link
Reference in a new issue