mirror of
https://github.com/thekiwismarthome/shopping-list-manager.git
synced 2026-05-01 11:46:30 +00:00
Merge pull request #6 from thekiwismarthome/2.3.0---Product-Barcode-Scanner-w/-OpenFoodFacts-Lookup
2.3.0 product barcode scanner w/ open food facts lookup
This commit is contained in:
@@ -171,6 +171,14 @@ async def _async_register_websocket_handlers(
|
|||||||
)
|
)
|
||||||
|
|
||||||
# Products handlers
|
# Products handlers
|
||||||
|
websocket_api.async_register_command(
|
||||||
|
hass,
|
||||||
|
handlers.websocket_download_product_image,
|
||||||
|
)
|
||||||
|
websocket_api.async_register_command(
|
||||||
|
hass,
|
||||||
|
handlers.websocket_search_by_barcode,
|
||||||
|
)
|
||||||
websocket_api.async_register_command(
|
websocket_api.async_register_command(
|
||||||
hass,
|
hass,
|
||||||
handlers.websocket_search_products,
|
handlers.websocket_search_products,
|
||||||
|
|||||||
@@ -1,14 +1,23 @@
|
|||||||
"""WebSocket API handlers for Shopping List Manager."""
|
"""WebSocket API handlers for Shopping List Manager."""
|
||||||
|
import io
|
||||||
import logging
|
import logging
|
||||||
|
import re
|
||||||
|
from pathlib import Path
|
||||||
from typing import Any, Dict
|
from typing import Any, Dict
|
||||||
|
|
||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
|
from aiohttp import ClientTimeout
|
||||||
|
from PIL import Image
|
||||||
|
|
||||||
from homeassistant.components import websocket_api
|
from homeassistant.components import websocket_api
|
||||||
from homeassistant.core import HomeAssistant, callback
|
from homeassistant.core import HomeAssistant, callback
|
||||||
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
||||||
from ..const import DOMAIN
|
from ..const import DOMAIN
|
||||||
|
|
||||||
from ..const import (
|
from ..const import (
|
||||||
|
IMAGE_SIZE,
|
||||||
|
IMAGE_QUALITY,
|
||||||
|
IMAGES_LOCAL_DIR,
|
||||||
WS_TYPE_LISTS_GET_ALL,
|
WS_TYPE_LISTS_GET_ALL,
|
||||||
WS_TYPE_LISTS_CREATE,
|
WS_TYPE_LISTS_CREATE,
|
||||||
WS_TYPE_LISTS_UPDATE,
|
WS_TYPE_LISTS_UPDATE,
|
||||||
@@ -782,6 +791,87 @@ def websocket_get_list_total(
|
|||||||
# PRODUCT HANDLERS
|
# PRODUCT HANDLERS
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
|
|
||||||
|
@websocket_api.websocket_command(
|
||||||
|
{
|
||||||
|
vol.Required("type"): "shopping_list_manager/products/download_image",
|
||||||
|
vol.Required("image_url"): str,
|
||||||
|
vol.Required("product_name"): str,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
@websocket_api.async_response
|
||||||
|
async def websocket_download_product_image(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
connection: websocket_api.ActiveConnection,
|
||||||
|
msg: Dict[str, Any],
|
||||||
|
) -> None:
|
||||||
|
"""Download a remote image and save it as WebP to the local images directory."""
|
||||||
|
raw_url: str = msg["image_url"]
|
||||||
|
product_name: str = msg["product_name"]
|
||||||
|
|
||||||
|
safe_stem = re.sub(r"[^a-z0-9_]", "", product_name.lower().replace(" ", "_")) or "product"
|
||||||
|
filename = f"{safe_stem}.webp"
|
||||||
|
images_dir = Path(hass.config.path(IMAGES_LOCAL_DIR))
|
||||||
|
images_dir.mkdir(parents=True, exist_ok=True)
|
||||||
|
dest = images_dir / filename
|
||||||
|
|
||||||
|
try:
|
||||||
|
session = async_get_clientsession(hass)
|
||||||
|
headers = {"User-Agent": "Mozilla/5.0 (compatible; HomeAssistant/ShoppingListManager)"}
|
||||||
|
async with session.get(raw_url, timeout=ClientTimeout(total=15), headers=headers) as resp:
|
||||||
|
if resp.status != 200:
|
||||||
|
connection.send_error(msg["id"], "download_failed", f"HTTP {resp.status}")
|
||||||
|
return
|
||||||
|
raw = await resp.read()
|
||||||
|
except Exception as exc: # noqa: BLE001
|
||||||
|
connection.send_error(msg["id"], "download_failed", str(exc))
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
img = Image.open(io.BytesIO(raw))
|
||||||
|
# Convert to RGB for reliable lossy WebP encoding
|
||||||
|
# (RGBA, palette, grayscale modes can fail or produce oversized files)
|
||||||
|
if img.mode == "RGBA":
|
||||||
|
bg = Image.new("RGB", img.size, (255, 255, 255))
|
||||||
|
bg.paste(img, mask=img.split()[3])
|
||||||
|
img = bg
|
||||||
|
elif img.mode != "RGB":
|
||||||
|
img = img.convert("RGB")
|
||||||
|
img.thumbnail((IMAGE_SIZE, IMAGE_SIZE), Image.LANCZOS)
|
||||||
|
out = io.BytesIO()
|
||||||
|
img.save(out, format="WEBP", quality=IMAGE_QUALITY)
|
||||||
|
dest.write_bytes(out.getvalue())
|
||||||
|
except Exception as exc: # noqa: BLE001
|
||||||
|
connection.send_error(msg["id"], "conversion_failed", str(exc))
|
||||||
|
return
|
||||||
|
|
||||||
|
connection.send_result(
|
||||||
|
msg["id"],
|
||||||
|
{"local_url": f"/local/shopping_list_manager/images/{filename}"},
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@websocket_api.websocket_command(
|
||||||
|
{
|
||||||
|
vol.Required("type"): "shopping_list_manager/products/search_by_barcode",
|
||||||
|
vol.Required("barcode"): str,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
@callback
|
||||||
|
def websocket_search_by_barcode(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
connection: websocket_api.ActiveConnection,
|
||||||
|
msg: Dict[str, Any],
|
||||||
|
) -> None:
|
||||||
|
"""Find a single product by exact barcode match."""
|
||||||
|
storage = get_storage(hass)
|
||||||
|
barcode = msg["barcode"].strip()
|
||||||
|
match = next(
|
||||||
|
(p for p in storage._products.values() if p.barcode and p.barcode == barcode),
|
||||||
|
None,
|
||||||
|
)
|
||||||
|
connection.send_result(msg["id"], {"product": match.to_dict() if match else None})
|
||||||
|
|
||||||
|
|
||||||
@websocket_api.websocket_command(
|
@websocket_api.websocket_command(
|
||||||
{
|
{
|
||||||
vol.Required("type"): "shopping_list_manager/products/substitutes",
|
vol.Required("type"): "shopping_list_manager/products/substitutes",
|
||||||
|
|||||||
Reference in New Issue
Block a user