mirror of
https://github.com/home-assistant/core.git
synced 2025-07-14 08:47:10 +00:00
Limit USB discovery to specific manufacturer/description/serial_number matches (#55236)
* Limit USB discovery to specific manufacturer/description/serial_number matches * test for None case
This commit is contained in:
parent
2d5176eee9
commit
a89057ece5
@ -2,6 +2,7 @@
|
|||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
import dataclasses
|
import dataclasses
|
||||||
|
import fnmatch
|
||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
@ -72,6 +73,13 @@ async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
|
|||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def _fnmatch_lower(name: str | None, pattern: str) -> bool:
|
||||||
|
"""Match a lowercase version of the name."""
|
||||||
|
if name is None:
|
||||||
|
return False
|
||||||
|
return fnmatch.fnmatch(name.lower(), pattern)
|
||||||
|
|
||||||
|
|
||||||
class USBDiscovery:
|
class USBDiscovery:
|
||||||
"""Manage USB Discovery."""
|
"""Manage USB Discovery."""
|
||||||
|
|
||||||
@ -152,6 +160,18 @@ class USBDiscovery:
|
|||||||
continue
|
continue
|
||||||
if "pid" in matcher and device.pid != matcher["pid"]:
|
if "pid" in matcher and device.pid != matcher["pid"]:
|
||||||
continue
|
continue
|
||||||
|
if "serial_number" in matcher and not _fnmatch_lower(
|
||||||
|
device.serial_number, matcher["serial_number"]
|
||||||
|
):
|
||||||
|
continue
|
||||||
|
if "manufacturer" in matcher and not _fnmatch_lower(
|
||||||
|
device.manufacturer, matcher["manufacturer"]
|
||||||
|
):
|
||||||
|
continue
|
||||||
|
if "description" in matcher and not _fnmatch_lower(
|
||||||
|
device.description, matcher["description"]
|
||||||
|
):
|
||||||
|
continue
|
||||||
flow: USBFlow = {
|
flow: USBFlow = {
|
||||||
"domain": matcher["domain"],
|
"domain": matcher["domain"],
|
||||||
"context": {"source": config_entries.SOURCE_USB},
|
"context": {"source": config_entries.SOURCE_USB},
|
||||||
|
@ -16,10 +16,9 @@
|
|||||||
"zigpy-znp==0.5.3"
|
"zigpy-znp==0.5.3"
|
||||||
],
|
],
|
||||||
"usb": [
|
"usb": [
|
||||||
{"vid":"10C4","pid":"EA60","known_devices":["slae.sh cc2652rb stick"]},
|
{"vid":"10C4","pid":"EA60","description":"*2652*","known_devices":["slae.sh cc2652rb stick"]},
|
||||||
{"vid":"1CF1","pid":"0030","known_devices":["Conbee II"]},
|
{"vid":"1CF1","pid":"0030","description":"*conbee*","known_devices":["Conbee II"]},
|
||||||
{"vid":"1A86","pid":"7523","known_devices":["Electrolama zig-a-zig-ah"]},
|
{"vid":"10C4","pid":"8A2A","description":"*zigbee*","known_devices":["Nortek HUSBZB-1"]}
|
||||||
{"vid":"10C4","pid":"8A2A","known_devices":["Nortek HUSBZB-1"]}
|
|
||||||
],
|
],
|
||||||
"codeowners": ["@dmulcahey", "@adminiuga"],
|
"codeowners": ["@dmulcahey", "@adminiuga"],
|
||||||
"zeroconf": [
|
"zeroconf": [
|
||||||
|
@ -9,22 +9,20 @@ USB = [
|
|||||||
{
|
{
|
||||||
"domain": "zha",
|
"domain": "zha",
|
||||||
"vid": "10C4",
|
"vid": "10C4",
|
||||||
"pid": "EA60"
|
"pid": "EA60",
|
||||||
|
"description": "*2652*"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"domain": "zha",
|
"domain": "zha",
|
||||||
"vid": "1CF1",
|
"vid": "1CF1",
|
||||||
"pid": "0030"
|
"pid": "0030",
|
||||||
},
|
"description": "*conbee*"
|
||||||
{
|
|
||||||
"domain": "zha",
|
|
||||||
"vid": "1A86",
|
|
||||||
"pid": "7523"
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"domain": "zha",
|
"domain": "zha",
|
||||||
"vid": "10C4",
|
"vid": "10C4",
|
||||||
"pid": "8A2A"
|
"pid": "8A2A",
|
||||||
|
"description": "*zigbee*"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"domain": "zwave_js",
|
"domain": "zwave_js",
|
||||||
|
@ -210,6 +210,9 @@ MANIFEST_SCHEMA = vol.Schema(
|
|||||||
{
|
{
|
||||||
vol.Optional("vid"): vol.All(str, verify_uppercase),
|
vol.Optional("vid"): vol.All(str, verify_uppercase),
|
||||||
vol.Optional("pid"): vol.All(str, verify_uppercase),
|
vol.Optional("pid"): vol.All(str, verify_uppercase),
|
||||||
|
vol.Optional("serial_number"): vol.All(str, verify_lowercase),
|
||||||
|
vol.Optional("manufacturer"): vol.All(str, verify_lowercase),
|
||||||
|
vol.Optional("description"): vol.All(str, verify_lowercase),
|
||||||
vol.Optional("known_devices"): [str],
|
vol.Optional("known_devices"): [str],
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
@ -9,7 +9,7 @@ from homeassistant.components import usb
|
|||||||
from homeassistant.const import EVENT_HOMEASSISTANT_STARTED
|
from homeassistant.const import EVENT_HOMEASSISTANT_STARTED
|
||||||
from homeassistant.setup import async_setup_component
|
from homeassistant.setup import async_setup_component
|
||||||
|
|
||||||
from . import slae_sh_device
|
from . import conbee_device, slae_sh_device
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(name="operating_system")
|
@pytest.fixture(name="operating_system")
|
||||||
@ -171,6 +171,297 @@ async def test_discovered_by_websocket_scan(hass, hass_ws_client):
|
|||||||
assert mock_config_flow.mock_calls[0][1][0] == "test1"
|
assert mock_config_flow.mock_calls[0][1][0] == "test1"
|
||||||
|
|
||||||
|
|
||||||
|
async def test_discovered_by_websocket_scan_limited_by_description_matcher(
|
||||||
|
hass, hass_ws_client
|
||||||
|
):
|
||||||
|
"""Test a device is discovered from websocket scan is limited by the description matcher."""
|
||||||
|
new_usb = [
|
||||||
|
{"domain": "test1", "vid": "3039", "pid": "3039", "description": "*2652*"}
|
||||||
|
]
|
||||||
|
|
||||||
|
mock_comports = [
|
||||||
|
MagicMock(
|
||||||
|
device=slae_sh_device.device,
|
||||||
|
vid=12345,
|
||||||
|
pid=12345,
|
||||||
|
serial_number=slae_sh_device.serial_number,
|
||||||
|
manufacturer=slae_sh_device.manufacturer,
|
||||||
|
description=slae_sh_device.description,
|
||||||
|
)
|
||||||
|
]
|
||||||
|
|
||||||
|
with patch("pyudev.Context", side_effect=ImportError), patch(
|
||||||
|
"homeassistant.components.usb.async_get_usb", return_value=new_usb
|
||||||
|
), patch(
|
||||||
|
"homeassistant.components.usb.comports", return_value=mock_comports
|
||||||
|
), patch.object(
|
||||||
|
hass.config_entries.flow, "async_init"
|
||||||
|
) as mock_config_flow:
|
||||||
|
assert await async_setup_component(hass, "usb", {"usb": {}})
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STARTED)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
ws_client = await hass_ws_client(hass)
|
||||||
|
await ws_client.send_json({"id": 1, "type": "usb/scan"})
|
||||||
|
response = await ws_client.receive_json()
|
||||||
|
assert response["success"]
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert len(mock_config_flow.mock_calls) == 1
|
||||||
|
assert mock_config_flow.mock_calls[0][1][0] == "test1"
|
||||||
|
|
||||||
|
|
||||||
|
async def test_discovered_by_websocket_scan_rejected_by_description_matcher(
|
||||||
|
hass, hass_ws_client
|
||||||
|
):
|
||||||
|
"""Test a device is discovered from websocket scan rejected by the description matcher."""
|
||||||
|
new_usb = [
|
||||||
|
{"domain": "test1", "vid": "3039", "pid": "3039", "description": "*not_it*"}
|
||||||
|
]
|
||||||
|
|
||||||
|
mock_comports = [
|
||||||
|
MagicMock(
|
||||||
|
device=slae_sh_device.device,
|
||||||
|
vid=12345,
|
||||||
|
pid=12345,
|
||||||
|
serial_number=slae_sh_device.serial_number,
|
||||||
|
manufacturer=slae_sh_device.manufacturer,
|
||||||
|
description=slae_sh_device.description,
|
||||||
|
)
|
||||||
|
]
|
||||||
|
|
||||||
|
with patch("pyudev.Context", side_effect=ImportError), patch(
|
||||||
|
"homeassistant.components.usb.async_get_usb", return_value=new_usb
|
||||||
|
), patch(
|
||||||
|
"homeassistant.components.usb.comports", return_value=mock_comports
|
||||||
|
), patch.object(
|
||||||
|
hass.config_entries.flow, "async_init"
|
||||||
|
) as mock_config_flow:
|
||||||
|
assert await async_setup_component(hass, "usb", {"usb": {}})
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STARTED)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
ws_client = await hass_ws_client(hass)
|
||||||
|
await ws_client.send_json({"id": 1, "type": "usb/scan"})
|
||||||
|
response = await ws_client.receive_json()
|
||||||
|
assert response["success"]
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert len(mock_config_flow.mock_calls) == 0
|
||||||
|
|
||||||
|
|
||||||
|
async def test_discovered_by_websocket_scan_limited_by_serial_number_matcher(
|
||||||
|
hass, hass_ws_client
|
||||||
|
):
|
||||||
|
"""Test a device is discovered from websocket scan is limited by the serial_number matcher."""
|
||||||
|
new_usb = [
|
||||||
|
{
|
||||||
|
"domain": "test1",
|
||||||
|
"vid": "3039",
|
||||||
|
"pid": "3039",
|
||||||
|
"serial_number": "00_12_4b_00*",
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
mock_comports = [
|
||||||
|
MagicMock(
|
||||||
|
device=slae_sh_device.device,
|
||||||
|
vid=12345,
|
||||||
|
pid=12345,
|
||||||
|
serial_number=slae_sh_device.serial_number,
|
||||||
|
manufacturer=slae_sh_device.manufacturer,
|
||||||
|
description=slae_sh_device.description,
|
||||||
|
)
|
||||||
|
]
|
||||||
|
|
||||||
|
with patch("pyudev.Context", side_effect=ImportError), patch(
|
||||||
|
"homeassistant.components.usb.async_get_usb", return_value=new_usb
|
||||||
|
), patch(
|
||||||
|
"homeassistant.components.usb.comports", return_value=mock_comports
|
||||||
|
), patch.object(
|
||||||
|
hass.config_entries.flow, "async_init"
|
||||||
|
) as mock_config_flow:
|
||||||
|
assert await async_setup_component(hass, "usb", {"usb": {}})
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STARTED)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
ws_client = await hass_ws_client(hass)
|
||||||
|
await ws_client.send_json({"id": 1, "type": "usb/scan"})
|
||||||
|
response = await ws_client.receive_json()
|
||||||
|
assert response["success"]
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert len(mock_config_flow.mock_calls) == 1
|
||||||
|
assert mock_config_flow.mock_calls[0][1][0] == "test1"
|
||||||
|
|
||||||
|
|
||||||
|
async def test_discovered_by_websocket_scan_rejected_by_serial_number_matcher(
|
||||||
|
hass, hass_ws_client
|
||||||
|
):
|
||||||
|
"""Test a device is discovered from websocket scan is rejected by the serial_number matcher."""
|
||||||
|
new_usb = [
|
||||||
|
{"domain": "test1", "vid": "3039", "pid": "3039", "serial_number": "123*"}
|
||||||
|
]
|
||||||
|
|
||||||
|
mock_comports = [
|
||||||
|
MagicMock(
|
||||||
|
device=slae_sh_device.device,
|
||||||
|
vid=12345,
|
||||||
|
pid=12345,
|
||||||
|
serial_number=slae_sh_device.serial_number,
|
||||||
|
manufacturer=slae_sh_device.manufacturer,
|
||||||
|
description=slae_sh_device.description,
|
||||||
|
)
|
||||||
|
]
|
||||||
|
|
||||||
|
with patch("pyudev.Context", side_effect=ImportError), patch(
|
||||||
|
"homeassistant.components.usb.async_get_usb", return_value=new_usb
|
||||||
|
), patch(
|
||||||
|
"homeassistant.components.usb.comports", return_value=mock_comports
|
||||||
|
), patch.object(
|
||||||
|
hass.config_entries.flow, "async_init"
|
||||||
|
) as mock_config_flow:
|
||||||
|
assert await async_setup_component(hass, "usb", {"usb": {}})
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STARTED)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
ws_client = await hass_ws_client(hass)
|
||||||
|
await ws_client.send_json({"id": 1, "type": "usb/scan"})
|
||||||
|
response = await ws_client.receive_json()
|
||||||
|
assert response["success"]
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert len(mock_config_flow.mock_calls) == 0
|
||||||
|
|
||||||
|
|
||||||
|
async def test_discovered_by_websocket_scan_limited_by_manufacturer_matcher(
|
||||||
|
hass, hass_ws_client
|
||||||
|
):
|
||||||
|
"""Test a device is discovered from websocket scan is limited by the manufacturer matcher."""
|
||||||
|
new_usb = [
|
||||||
|
{
|
||||||
|
"domain": "test1",
|
||||||
|
"vid": "3039",
|
||||||
|
"pid": "3039",
|
||||||
|
"manufacturer": "dresden elektronik ingenieurtechnik*",
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
mock_comports = [
|
||||||
|
MagicMock(
|
||||||
|
device=conbee_device.device,
|
||||||
|
vid=12345,
|
||||||
|
pid=12345,
|
||||||
|
serial_number=conbee_device.serial_number,
|
||||||
|
manufacturer=conbee_device.manufacturer,
|
||||||
|
description=conbee_device.description,
|
||||||
|
)
|
||||||
|
]
|
||||||
|
|
||||||
|
with patch("pyudev.Context", side_effect=ImportError), patch(
|
||||||
|
"homeassistant.components.usb.async_get_usb", return_value=new_usb
|
||||||
|
), patch(
|
||||||
|
"homeassistant.components.usb.comports", return_value=mock_comports
|
||||||
|
), patch.object(
|
||||||
|
hass.config_entries.flow, "async_init"
|
||||||
|
) as mock_config_flow:
|
||||||
|
assert await async_setup_component(hass, "usb", {"usb": {}})
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STARTED)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
ws_client = await hass_ws_client(hass)
|
||||||
|
await ws_client.send_json({"id": 1, "type": "usb/scan"})
|
||||||
|
response = await ws_client.receive_json()
|
||||||
|
assert response["success"]
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert len(mock_config_flow.mock_calls) == 1
|
||||||
|
assert mock_config_flow.mock_calls[0][1][0] == "test1"
|
||||||
|
|
||||||
|
|
||||||
|
async def test_discovered_by_websocket_scan_rejected_by_manufacturer_matcher(
|
||||||
|
hass, hass_ws_client
|
||||||
|
):
|
||||||
|
"""Test a device is discovered from websocket scan is rejected by the manufacturer matcher."""
|
||||||
|
new_usb = [
|
||||||
|
{
|
||||||
|
"domain": "test1",
|
||||||
|
"vid": "3039",
|
||||||
|
"pid": "3039",
|
||||||
|
"manufacturer": "other vendor*",
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
mock_comports = [
|
||||||
|
MagicMock(
|
||||||
|
device=conbee_device.device,
|
||||||
|
vid=12345,
|
||||||
|
pid=12345,
|
||||||
|
serial_number=conbee_device.serial_number,
|
||||||
|
manufacturer=conbee_device.manufacturer,
|
||||||
|
description=conbee_device.description,
|
||||||
|
)
|
||||||
|
]
|
||||||
|
|
||||||
|
with patch("pyudev.Context", side_effect=ImportError), patch(
|
||||||
|
"homeassistant.components.usb.async_get_usb", return_value=new_usb
|
||||||
|
), patch(
|
||||||
|
"homeassistant.components.usb.comports", return_value=mock_comports
|
||||||
|
), patch.object(
|
||||||
|
hass.config_entries.flow, "async_init"
|
||||||
|
) as mock_config_flow:
|
||||||
|
assert await async_setup_component(hass, "usb", {"usb": {}})
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STARTED)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
ws_client = await hass_ws_client(hass)
|
||||||
|
await ws_client.send_json({"id": 1, "type": "usb/scan"})
|
||||||
|
response = await ws_client.receive_json()
|
||||||
|
assert response["success"]
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert len(mock_config_flow.mock_calls) == 0
|
||||||
|
|
||||||
|
|
||||||
|
async def test_discovered_by_websocket_rejected_with_empty_serial_number_only(
|
||||||
|
hass, hass_ws_client
|
||||||
|
):
|
||||||
|
"""Test a device is discovered from websocket is rejected with empty serial number."""
|
||||||
|
new_usb = [
|
||||||
|
{"domain": "test1", "vid": "3039", "pid": "3039", "serial_number": "123*"}
|
||||||
|
]
|
||||||
|
|
||||||
|
mock_comports = [
|
||||||
|
MagicMock(
|
||||||
|
device=conbee_device.device,
|
||||||
|
vid=12345,
|
||||||
|
pid=12345,
|
||||||
|
serial_number=None,
|
||||||
|
manufacturer=None,
|
||||||
|
description=None,
|
||||||
|
)
|
||||||
|
]
|
||||||
|
|
||||||
|
with patch("pyudev.Context", side_effect=ImportError), patch(
|
||||||
|
"homeassistant.components.usb.async_get_usb", return_value=new_usb
|
||||||
|
), patch(
|
||||||
|
"homeassistant.components.usb.comports", return_value=mock_comports
|
||||||
|
), patch.object(
|
||||||
|
hass.config_entries.flow, "async_init"
|
||||||
|
) as mock_config_flow:
|
||||||
|
assert await async_setup_component(hass, "usb", {"usb": {}})
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STARTED)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
ws_client = await hass_ws_client(hass)
|
||||||
|
await ws_client.send_json({"id": 1, "type": "usb/scan"})
|
||||||
|
response = await ws_client.receive_json()
|
||||||
|
assert response["success"]
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert len(mock_config_flow.mock_calls) == 0
|
||||||
|
|
||||||
|
|
||||||
async def test_discovered_by_websocket_scan_match_vid_only(hass, hass_ws_client):
|
async def test_discovered_by_websocket_scan_match_vid_only(hass, hass_ws_client):
|
||||||
"""Test a device is discovered from websocket scan only matching vid."""
|
"""Test a device is discovered from websocket scan only matching vid."""
|
||||||
new_usb = [{"domain": "test1", "vid": "3039"}]
|
new_usb = [{"domain": "test1", "vid": "3039"}]
|
||||||
|
Loading…
x
Reference in New Issue
Block a user