2022-08-11 12:56:28 +02:00
|
|
|
"""Support for the Mikrotik Router update service."""
|
2024-04-12 14:39:36 +03:00
|
|
|
|
2023-08-09 09:53:48 +02:00
|
|
|
from __future__ import annotations
|
2022-08-11 12:56:28 +02:00
|
|
|
|
2024-04-16 18:31:35 +03:00
|
|
|
import asyncio
|
2023-08-09 09:53:48 +02:00
|
|
|
from logging import getLogger
|
|
|
|
from typing import Any
|
|
|
|
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2024-04-12 16:13:54 +03:00
|
|
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
2023-08-09 09:53:48 +02:00
|
|
|
|
2022-08-11 12:56:28 +02:00
|
|
|
from homeassistant.components.update import (
|
|
|
|
UpdateEntity,
|
|
|
|
UpdateDeviceClass,
|
|
|
|
UpdateEntityFeature,
|
|
|
|
)
|
2023-08-09 09:53:48 +02:00
|
|
|
|
|
|
|
from .coordinator import MikrotikCoordinator
|
|
|
|
from .entity import MikrotikEntity, async_add_entities
|
2022-08-11 12:56:28 +02:00
|
|
|
from .update_types import (
|
|
|
|
SENSOR_TYPES,
|
|
|
|
SENSOR_SERVICES,
|
|
|
|
)
|
2024-04-16 18:31:35 +03:00
|
|
|
from packaging.version import Version
|
2022-08-11 12:56:28 +02:00
|
|
|
|
2023-08-09 09:53:48 +02:00
|
|
|
_LOGGER = getLogger(__name__)
|
2022-08-11 12:56:28 +02:00
|
|
|
DEVICE_UPDATE = "device_update"
|
|
|
|
|
|
|
|
|
|
|
|
# ---------------------------
|
|
|
|
# async_setup_entry
|
|
|
|
# ---------------------------
|
2023-08-09 09:53:48 +02:00
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config_entry: ConfigEntry,
|
|
|
|
_async_add_entities: AddEntitiesCallback,
|
|
|
|
) -> None:
|
2022-08-11 12:56:28 +02:00
|
|
|
"""Set up entry for component"""
|
|
|
|
dispatcher = {
|
2022-08-11 14:58:30 +02:00
|
|
|
"MikrotikRouterOSUpdate": MikrotikRouterOSUpdate,
|
|
|
|
"MikrotikRouterBoardFWUpdate": MikrotikRouterBoardFWUpdate,
|
2022-08-11 12:56:28 +02:00
|
|
|
}
|
2023-08-09 09:53:48 +02:00
|
|
|
await async_add_entities(hass, config_entry, dispatcher)
|
2022-08-11 12:56:28 +02:00
|
|
|
|
|
|
|
|
|
|
|
# ---------------------------
|
2022-08-11 14:58:30 +02:00
|
|
|
# MikrotikRouterOSUpdate
|
2022-08-11 12:56:28 +02:00
|
|
|
# ---------------------------
|
2022-08-11 14:58:30 +02:00
|
|
|
class MikrotikRouterOSUpdate(MikrotikEntity, UpdateEntity):
|
2022-08-11 12:56:28 +02:00
|
|
|
"""Define an Mikrotik Controller Update entity."""
|
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self,
|
2023-08-09 09:53:48 +02:00
|
|
|
coordinator: MikrotikCoordinator,
|
2022-08-11 12:56:28 +02:00
|
|
|
entity_description,
|
2023-08-09 09:53:48 +02:00
|
|
|
uid: str | None = None,
|
2022-08-11 12:56:28 +02:00
|
|
|
):
|
|
|
|
"""Set up device update entity."""
|
2023-08-09 09:53:48 +02:00
|
|
|
super().__init__(coordinator, entity_description, uid)
|
2022-08-11 12:56:28 +02:00
|
|
|
|
|
|
|
self._attr_supported_features = UpdateEntityFeature.INSTALL
|
|
|
|
self._attr_supported_features |= UpdateEntityFeature.BACKUP
|
|
|
|
self._attr_supported_features |= UpdateEntityFeature.RELEASE_NOTES
|
2022-08-17 16:23:41 +02:00
|
|
|
self._attr_title = self.entity_description.title
|
2022-08-11 12:56:28 +02:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self) -> bool:
|
|
|
|
"""Return true if device is on."""
|
|
|
|
return self._data[self.entity_description.data_attribute]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def installed_version(self) -> str:
|
|
|
|
"""Version installed and in use."""
|
|
|
|
return self._data["installed-version"]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def latest_version(self) -> str:
|
|
|
|
"""Latest version available for install."""
|
|
|
|
return self._data["latest-version"]
|
|
|
|
|
|
|
|
async def options_updated(self) -> None:
|
|
|
|
"""No action needed."""
|
|
|
|
|
|
|
|
async def async_install(self, version: str, backup: bool, **kwargs: Any) -> None:
|
|
|
|
"""Install an update."""
|
|
|
|
if backup:
|
2023-08-09 09:53:48 +02:00
|
|
|
self.coordinator.execute("/system/backup", "save", None, None)
|
2022-08-11 12:56:28 +02:00
|
|
|
|
2023-08-09 09:53:48 +02:00
|
|
|
self.coordinator.execute("/system/package/update", "install", None, None)
|
2022-08-11 12:56:28 +02:00
|
|
|
|
|
|
|
async def async_release_notes(self) -> str:
|
|
|
|
"""Return the release notes."""
|
|
|
|
try:
|
2024-04-12 16:13:54 +03:00
|
|
|
session = async_get_clientsession(self.hass)
|
2024-04-16 18:31:35 +03:00
|
|
|
"""Get concatenated changelogs from installed_version to latest_version in reverse order."""
|
|
|
|
versions_to_fetch = generate_version_list(
|
|
|
|
self._data["installed-version"], self._data["latest-version"]
|
|
|
|
)
|
|
|
|
|
|
|
|
tasks = [fetch_changelog(session, version) for version in versions_to_fetch]
|
|
|
|
changelogs = await asyncio.gather(*tasks)
|
|
|
|
|
|
|
|
# Combine all non-empty changelogs, maintaining reverse order
|
2024-04-19 20:18:08 +03:00
|
|
|
combined_changelogs = "\n\n".join(filter(None, changelogs))
|
2024-04-16 18:31:35 +03:00
|
|
|
return combined_changelogs.replace("*) ", "- ")
|
|
|
|
|
2022-08-11 14:57:29 +02:00
|
|
|
except Exception as e:
|
|
|
|
_LOGGER.warning("Failed to download release notes (%s)", e)
|
2022-08-11 12:56:28 +02:00
|
|
|
|
2024-04-12 15:27:52 +03:00
|
|
|
return "Error fetching release notes."
|
2022-08-11 12:56:28 +02:00
|
|
|
|
|
|
|
@property
|
|
|
|
def release_url(self) -> str:
|
|
|
|
"""URL to the full release notes of the latest version available."""
|
|
|
|
return "https://mikrotik.com/download/changelogs"
|
2022-08-11 14:58:30 +02:00
|
|
|
|
|
|
|
|
|
|
|
# ---------------------------
|
|
|
|
# MikrotikRouterBoardFWUpdate
|
|
|
|
# ---------------------------
|
|
|
|
class MikrotikRouterBoardFWUpdate(MikrotikEntity, UpdateEntity):
|
|
|
|
"""Define an Mikrotik Controller Update entity."""
|
|
|
|
|
2022-08-17 12:54:26 +02:00
|
|
|
TYPE = DEVICE_UPDATE
|
|
|
|
_attr_device_class = UpdateDeviceClass.FIRMWARE
|
2022-08-17 13:09:59 +02:00
|
|
|
|
2022-08-11 14:58:30 +02:00
|
|
|
def __init__(
|
|
|
|
self,
|
2023-08-09 10:18:46 +02:00
|
|
|
coordinator: MikrotikCoordinator,
|
2022-08-11 14:58:30 +02:00
|
|
|
entity_description,
|
2023-08-09 10:18:46 +02:00
|
|
|
uid: str | None = None,
|
2022-08-11 14:58:30 +02:00
|
|
|
):
|
|
|
|
"""Set up device update entity."""
|
2023-08-09 10:18:46 +02:00
|
|
|
super().__init__(coordinator, entity_description, uid)
|
2022-08-11 14:58:30 +02:00
|
|
|
|
|
|
|
self._attr_supported_features = UpdateEntityFeature.INSTALL
|
2022-08-17 16:23:41 +02:00
|
|
|
self._attr_title = self.entity_description.title
|
2022-08-11 14:58:30 +02:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self) -> bool:
|
|
|
|
"""Return true if device is on."""
|
|
|
|
return (
|
|
|
|
self.data["routerboard"]["current-firmware"]
|
|
|
|
!= self.data["routerboard"]["upgrade-firmware"]
|
|
|
|
)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def installed_version(self) -> str:
|
|
|
|
"""Version installed and in use."""
|
|
|
|
return self._data["current-firmware"]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def latest_version(self) -> str:
|
|
|
|
"""Latest version available for install."""
|
|
|
|
return self._data["upgrade-firmware"]
|
|
|
|
|
|
|
|
async def options_updated(self) -> None:
|
|
|
|
"""No action needed."""
|
|
|
|
|
|
|
|
async def async_install(self, version: str, backup: bool, **kwargs: Any) -> None:
|
|
|
|
"""Install an update."""
|
2023-08-09 09:53:48 +02:00
|
|
|
self.coordinator.execute("/system/routerboard", "upgrade", None, None)
|
|
|
|
self.coordinator.execute("/system", "reboot", None, None)
|
2024-04-16 18:31:35 +03:00
|
|
|
|
|
|
|
|
|
|
|
async def fetch_changelog(session, version: str) -> str:
|
|
|
|
"""Asynchronously fetch the changelog for a given version."""
|
|
|
|
url = f"https://cdn.mikrotik.com/routeros/{version}/CHANGELOG"
|
|
|
|
try:
|
|
|
|
async with session.get(url) as response:
|
|
|
|
if response.status == 200:
|
|
|
|
text = await response.text()
|
|
|
|
return text.replace("*) ", "- ")
|
|
|
|
except Exception as e:
|
|
|
|
pass
|
|
|
|
return ""
|
|
|
|
|
|
|
|
|
|
|
|
def generate_version_list(start_version: str, end_version: str) -> list:
|
|
|
|
"""Generate a list of version strings from start_version to end_version in reverse order."""
|
|
|
|
start = Version(start_version)
|
|
|
|
end = Version(end_version)
|
|
|
|
versions = []
|
|
|
|
|
|
|
|
current = end
|
|
|
|
while current >= start:
|
|
|
|
versions.append(str(current))
|
|
|
|
current = decrement_version(current, start)
|
|
|
|
|
|
|
|
return versions
|
|
|
|
|
|
|
|
|
|
|
|
def decrement_version(version: Version, start_version: Version) -> Version:
|
|
|
|
"""Decrement version by the smallest possible step without going below start_version."""
|
|
|
|
if version.micro > 0:
|
|
|
|
next_patch = version.micro - 1
|
|
|
|
return Version(f"{version.major}.{version.minor}.{next_patch}")
|
|
|
|
elif version.minor > 0:
|
|
|
|
next_minor = version.minor - 1
|
|
|
|
return Version(
|
|
|
|
f"{version.major}.{next_minor}.999"
|
|
|
|
) # Assuming .999 as max patch version
|
|
|
|
else:
|
|
|
|
next_major = version.major - 1
|
|
|
|
return Version(
|
|
|
|
f"{next_major}.999.999"
|
|
|
|
) # Assuming .999 as max minor and patch version
|