2024-05-31 12:44:28 +00:00
|
|
|
"""Support for displaying collected data over SNMP."""
|
|
|
|
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
import logging
|
|
|
|
|
|
|
|
from pysnmp.hlapi.asyncio import (
|
|
|
|
CommunityData,
|
|
|
|
ContextData,
|
|
|
|
ObjectIdentity,
|
|
|
|
ObjectType,
|
|
|
|
SnmpEngine,
|
|
|
|
Udp6TransportTarget,
|
|
|
|
UdpTransportTarget,
|
|
|
|
UsmUserData,
|
|
|
|
)
|
|
|
|
from pysnmp.hlapi.asyncio.cmdgen import lcd, vbProcessor
|
|
|
|
from pysnmp.smi.builder import MibBuilder
|
|
|
|
|
|
|
|
from homeassistant.const import EVENT_HOMEASSISTANT_STOP
|
|
|
|
from homeassistant.core import Event, HomeAssistant, callback
|
|
|
|
from homeassistant.helpers.singleton import singleton
|
|
|
|
|
|
|
|
DATA_SNMP_ENGINE = "snmp_engine"
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2024-06-07 18:09:48 +00:00
|
|
|
type CommandArgsType = tuple[
|
|
|
|
SnmpEngine,
|
|
|
|
UsmUserData | CommunityData,
|
|
|
|
UdpTransportTarget | Udp6TransportTarget,
|
|
|
|
ContextData,
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2024-05-31 12:44:28 +00:00
|
|
|
type RequestArgsType = tuple[
|
|
|
|
SnmpEngine,
|
|
|
|
UsmUserData | CommunityData,
|
|
|
|
UdpTransportTarget | Udp6TransportTarget,
|
|
|
|
ContextData,
|
|
|
|
ObjectType,
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2024-06-07 18:09:48 +00:00
|
|
|
async def async_create_command_cmd_args(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
auth_data: UsmUserData | CommunityData,
|
|
|
|
target: UdpTransportTarget | Udp6TransportTarget,
|
|
|
|
) -> CommandArgsType:
|
|
|
|
"""Create command arguments.
|
|
|
|
|
|
|
|
The ObjectType needs to be created dynamically by the caller.
|
|
|
|
"""
|
|
|
|
engine = await async_get_snmp_engine(hass)
|
|
|
|
return (engine, auth_data, target, ContextData())
|
|
|
|
|
|
|
|
|
2024-05-31 12:44:28 +00:00
|
|
|
async def async_create_request_cmd_args(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
auth_data: UsmUserData | CommunityData,
|
|
|
|
target: UdpTransportTarget | Udp6TransportTarget,
|
|
|
|
object_id: str,
|
|
|
|
) -> RequestArgsType:
|
2024-06-07 18:09:48 +00:00
|
|
|
"""Create request arguments.
|
|
|
|
|
|
|
|
The same ObjectType is used for all requests.
|
|
|
|
"""
|
|
|
|
engine, auth_data, target, context_data = await async_create_command_cmd_args(
|
|
|
|
hass, auth_data, target
|
2024-05-31 12:44:28 +00:00
|
|
|
)
|
2024-06-07 18:09:48 +00:00
|
|
|
object_type = ObjectType(ObjectIdentity(object_id))
|
|
|
|
return (engine, auth_data, target, context_data, object_type)
|
2024-05-31 12:44:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
@singleton(DATA_SNMP_ENGINE)
|
|
|
|
async def async_get_snmp_engine(hass: HomeAssistant) -> SnmpEngine:
|
|
|
|
"""Get the SNMP engine."""
|
|
|
|
engine = await hass.async_add_executor_job(_get_snmp_engine)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_shutdown_listener(ev: Event) -> None:
|
|
|
|
_LOGGER.debug("Unconfiguring SNMP engine")
|
|
|
|
lcd.unconfigure(engine, None)
|
|
|
|
|
|
|
|
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, _async_shutdown_listener)
|
|
|
|
return engine
|
|
|
|
|
|
|
|
|
|
|
|
def _get_snmp_engine() -> SnmpEngine:
|
|
|
|
"""Return a cached instance of SnmpEngine."""
|
|
|
|
engine = SnmpEngine()
|
|
|
|
mib_controller = vbProcessor.getMibViewController(engine)
|
|
|
|
# Actually load the MIBs from disk so we do
|
|
|
|
# not do it in the event loop
|
|
|
|
builder: MibBuilder = mib_controller.mibBuilder
|
|
|
|
if "PYSNMP-MIB" not in builder.mibSymbols:
|
|
|
|
builder.loadModules()
|
|
|
|
return engine
|