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
|
||||
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(
|
||||
hass,
|
||||
handlers.websocket_search_products,
|
||||
|
||||
@@ -1,14 +1,23 @@
|
||||
"""WebSocket API handlers for Shopping List Manager."""
|
||||
import io
|
||||
import logging
|
||||
import re
|
||||
from pathlib import Path
|
||||
from typing import Any, Dict
|
||||
|
||||
import voluptuous as vol
|
||||
from aiohttp import ClientTimeout
|
||||
from PIL import Image
|
||||
|
||||
from homeassistant.components import websocket_api
|
||||
from homeassistant.core import HomeAssistant, callback
|
||||
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
||||
from ..const import DOMAIN
|
||||
|
||||
from ..const import (
|
||||
IMAGE_SIZE,
|
||||
IMAGE_QUALITY,
|
||||
IMAGES_LOCAL_DIR,
|
||||
WS_TYPE_LISTS_GET_ALL,
|
||||
WS_TYPE_LISTS_CREATE,
|
||||
WS_TYPE_LISTS_UPDATE,
|
||||
@@ -782,6 +791,87 @@ def websocket_get_list_total(
|
||||
# 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(
|
||||
{
|
||||
vol.Required("type"): "shopping_list_manager/products/substitutes",
|
||||
|
||||
Reference in New Issue
Block a user