mirror of
https://github.com/home-assistant/supervisor.git
synced 2025-10-13 05:39:42 +00:00

* Refactory code / object handling * Next step * fix lint * Step 2 * Cleanup API code * cleanup addons code * cleanup data handling * Cleanup addons data handling * Cleanup docker api * clean docker api p2 * next cleanup round * cleanup start on snapshots * update format strings * fix setup * fix lint * fix lint * fix lint * fix tox * Fix wrong import of datetime module * Fix bug with attributes * fix extraction * Update core * Update logs * Expand scheduler * add support for time interval objects * next updates on tasks * Fix some things * Cleanup code / supervisor * fix lint * Fix some code styles * rename stuff * cleanup api call reload * fix lock replacment * fix lint * fix lint * fix bug * fix wrong config links * fix bugs * fix bug * Update version on startup * Fix some bugs * fix bug * Fix snapshot * Add wait boot options * fix lint * fix default config * fix snapshot * fix snapshot * load snapshots on startup * add log message at the end * Some cleanups * fix bug * add logger * add logger for supervisor update * Add more logger
133 lines
4.1 KiB
Python
133 lines
4.1 KiB
Python
"""Init file for HassIO addons."""
|
|
import asyncio
|
|
import logging
|
|
|
|
from .addon import Addon
|
|
from .repository import Repository
|
|
from .data import Data
|
|
from ..const import REPOSITORY_CORE, REPOSITORY_LOCAL, BOOT_AUTO
|
|
from ..coresys import CoreSysAttributes
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
BUILTIN_REPOSITORIES = set((REPOSITORY_CORE, REPOSITORY_LOCAL))
|
|
|
|
|
|
class AddonManager(CoreSysAttributes):
|
|
"""Manage addons inside HassIO."""
|
|
|
|
def __init__(self, coresys):
|
|
"""Initialize docker base wrapper."""
|
|
self.coresys = coresys
|
|
self.data = Data(coresys)
|
|
self.addons_obj = {}
|
|
self.repositories_obj = {}
|
|
|
|
@property
|
|
def list_addons(self):
|
|
"""Return a list of all addons."""
|
|
return list(self.addons_obj.values())
|
|
|
|
@property
|
|
def list_repositories(self):
|
|
"""Return list of addon repositories."""
|
|
return list(self.repositories_obj.values())
|
|
|
|
def get(self, addon_slug):
|
|
"""Return a adddon from slug."""
|
|
return self.addons_obj.get(addon_slug)
|
|
|
|
async def load(self):
|
|
"""Startup addon management."""
|
|
self.data.reload()
|
|
|
|
# init hassio built-in repositories
|
|
repositories = \
|
|
set(self._config.addons_repositories) | BUILTIN_REPOSITORIES
|
|
|
|
# init custom repositories & load addons
|
|
await self.load_repositories(repositories)
|
|
|
|
async def reload(self):
|
|
"""Update addons from repo and reload list."""
|
|
tasks = [repository.update() for repository in
|
|
self.repositories_obj.values()]
|
|
if tasks:
|
|
await asyncio.wait(tasks, loop=self._loop)
|
|
|
|
# read data from repositories
|
|
self.data.reload()
|
|
|
|
# update addons
|
|
await self.load_addons()
|
|
|
|
async def load_repositories(self, list_repositories):
|
|
"""Add a new custom repository."""
|
|
new_rep = set(list_repositories)
|
|
old_rep = set(self.repositories_obj)
|
|
|
|
# add new repository
|
|
async def _add_repository(url):
|
|
"""Helper function to async add repository."""
|
|
repository = Repository(self.coresys, url)
|
|
if not await repository.load():
|
|
_LOGGER.error("Can't load from repository %s", url)
|
|
return
|
|
self.repositories_obj[url] = repository
|
|
|
|
# don't add built-in repository to config
|
|
if url not in BUILTIN_REPOSITORIES:
|
|
self._config.add_addon_repository(url)
|
|
|
|
tasks = [_add_repository(url) for url in new_rep - old_rep]
|
|
if tasks:
|
|
await asyncio.wait(tasks, loop=self._loop)
|
|
|
|
# del new repository
|
|
for url in old_rep - new_rep - BUILTIN_REPOSITORIES:
|
|
self.repositories_obj.pop(url).remove()
|
|
self._config.drop_addon_repository(url)
|
|
|
|
# update data
|
|
self.data.reload()
|
|
await self.load_addons()
|
|
|
|
async def load_addons(self):
|
|
"""Update/add internal addon store."""
|
|
all_addons = set(self.data.system) | set(self.data.cache)
|
|
|
|
# calc diff
|
|
add_addons = all_addons - set(self.addons_obj)
|
|
del_addons = set(self.addons_obj) - all_addons
|
|
|
|
_LOGGER.info("Load addons: %d all - %d new - %d remove",
|
|
len(all_addons), len(add_addons), len(del_addons))
|
|
|
|
# new addons
|
|
tasks = []
|
|
for addon_slug in add_addons:
|
|
addon = Addon(self.coresys, addon_slug)
|
|
|
|
tasks.append(addon.load())
|
|
self.addons_obj[addon_slug] = addon
|
|
|
|
if tasks:
|
|
await asyncio.wait(tasks, loop=self._loop)
|
|
|
|
# remove
|
|
for addon_slug in del_addons:
|
|
self.addons_obj.pop(addon_slug)
|
|
|
|
async def auto_boot(self, stage):
|
|
"""Boot addons with mode auto."""
|
|
tasks = []
|
|
for addon in self.addons_obj.values():
|
|
if addon.is_installed and addon.boot == BOOT_AUTO and \
|
|
addon.startup == stage:
|
|
tasks.append(addon.start())
|
|
|
|
_LOGGER.info("Startup %s run %d addons", stage, len(tasks))
|
|
if tasks:
|
|
await asyncio.wait(tasks, loop=self._loop)
|
|
await asyncio.sleep(self._config.wait_boot, loop=self._loop)
|