2020-02-14 02:52:51 +00:00
|
|
|
"""Initialize the HACS base."""
|
|
|
|
# pylint: disable=unused-argument, bad-continuation
|
|
|
|
import json
|
|
|
|
import uuid
|
|
|
|
from datetime import timedelta
|
|
|
|
|
|
|
|
from homeassistant.helpers.event import async_call_later, async_track_time_interval
|
|
|
|
|
2020-05-21 22:48:00 +00:00
|
|
|
from aiogithubapi import AIOGitHubAPIException, AIOGitHubAPIRatelimitException
|
2020-02-14 02:52:51 +00:00
|
|
|
from integrationhelper import Logger
|
2020-05-03 20:23:17 +00:00
|
|
|
from queueman import QueueManager
|
2020-02-14 02:52:51 +00:00
|
|
|
|
2020-04-10 01:29:27 +00:00
|
|
|
from custom_components.hacs.hacsbase.task_factory import HacsTaskFactory
|
|
|
|
from custom_components.hacs.hacsbase.exceptions import HacsException
|
2020-02-14 02:52:51 +00:00
|
|
|
|
2020-04-10 01:29:27 +00:00
|
|
|
from custom_components.hacs.const import ELEMENT_TYPES
|
|
|
|
from custom_components.hacs.setup import setup_extra_stores
|
|
|
|
from custom_components.hacs.store import async_load_from_store, async_save_to_store
|
|
|
|
from custom_components.hacs.helpers.get_defaults import (
|
|
|
|
get_default_repos_lists,
|
|
|
|
get_default_repos_orgs,
|
|
|
|
)
|
|
|
|
|
|
|
|
from custom_components.hacs.helpers.register_repository import register_repository
|
2020-05-03 20:23:17 +00:00
|
|
|
from custom_components.hacs.helpers.remaining_github_calls import get_fetch_updates_for
|
2020-04-10 01:29:27 +00:00
|
|
|
from custom_components.hacs.globals import removed_repositories, get_removed, is_removed
|
|
|
|
from custom_components.hacs.repositories.removed import RemovedRepository
|
2020-02-14 02:52:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
class HacsStatus:
|
|
|
|
"""HacsStatus."""
|
|
|
|
|
|
|
|
startup = True
|
|
|
|
new = False
|
|
|
|
background_task = False
|
|
|
|
reloading_data = False
|
|
|
|
upgrading_all = False
|
|
|
|
|
|
|
|
|
|
|
|
class HacsFrontend:
|
|
|
|
"""HacsFrontend."""
|
|
|
|
|
|
|
|
version_running = None
|
|
|
|
version_available = None
|
|
|
|
update_pending = False
|
|
|
|
|
|
|
|
|
|
|
|
class HacsCommon:
|
|
|
|
"""Common for HACS."""
|
|
|
|
|
|
|
|
categories = []
|
|
|
|
default = []
|
|
|
|
installed = []
|
|
|
|
skip = []
|
|
|
|
|
|
|
|
|
|
|
|
class System:
|
|
|
|
"""System info."""
|
|
|
|
|
|
|
|
status = HacsStatus()
|
|
|
|
config_path = None
|
|
|
|
ha_version = None
|
|
|
|
disabled = False
|
|
|
|
lovelace_mode = "storage"
|
|
|
|
|
|
|
|
|
|
|
|
class Developer:
|
|
|
|
"""Developer settings/tools."""
|
|
|
|
|
|
|
|
template_id = "Repository ID"
|
|
|
|
template_content = ""
|
|
|
|
template_raw = ""
|
|
|
|
|
|
|
|
@property
|
|
|
|
def devcontainer(self):
|
|
|
|
"""Is it a devcontainer?"""
|
|
|
|
import os
|
|
|
|
|
|
|
|
if "DEVCONTAINER" in os.environ:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
class Hacs:
|
|
|
|
"""The base class of HACS, nested thoughout the project."""
|
|
|
|
|
|
|
|
token = f"{str(uuid.uuid4())}-{str(uuid.uuid4())}"
|
2020-05-03 20:23:17 +00:00
|
|
|
action = False
|
2020-02-14 02:52:51 +00:00
|
|
|
hacsweb = f"/hacsweb/{token}"
|
|
|
|
hacsapi = f"/hacsapi/{token}"
|
|
|
|
repositories = []
|
|
|
|
frontend = HacsFrontend()
|
|
|
|
repo = None
|
|
|
|
data_repo = None
|
|
|
|
developer = Developer()
|
|
|
|
data = None
|
|
|
|
configuration = None
|
|
|
|
logger = Logger("hacs")
|
|
|
|
github = None
|
|
|
|
hass = None
|
|
|
|
version = None
|
2020-04-10 01:29:27 +00:00
|
|
|
session = None
|
2020-02-14 02:52:51 +00:00
|
|
|
factory = HacsTaskFactory()
|
2020-05-03 20:23:17 +00:00
|
|
|
queue = QueueManager()
|
2020-02-14 02:52:51 +00:00
|
|
|
system = System()
|
|
|
|
recuring_tasks = []
|
|
|
|
common = HacsCommon()
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def init(hass, github_token):
|
|
|
|
"""Return a initialized HACS object."""
|
|
|
|
return Hacs()
|
|
|
|
|
|
|
|
def get_by_id(self, repository_id):
|
|
|
|
"""Get repository by ID."""
|
|
|
|
try:
|
|
|
|
for repository in self.repositories:
|
2020-05-21 22:48:00 +00:00
|
|
|
if str(repository.data.id) == str(repository_id):
|
2020-02-14 02:52:51 +00:00
|
|
|
return repository
|
|
|
|
except Exception: # pylint: disable=broad-except
|
|
|
|
pass
|
|
|
|
return None
|
|
|
|
|
|
|
|
def get_by_name(self, repository_full_name):
|
|
|
|
"""Get repository by full_name."""
|
|
|
|
try:
|
|
|
|
for repository in self.repositories:
|
2020-04-10 01:29:27 +00:00
|
|
|
if repository.data.full_name.lower() == repository_full_name.lower():
|
2020-02-14 02:52:51 +00:00
|
|
|
return repository
|
|
|
|
except Exception: # pylint: disable=broad-except
|
|
|
|
pass
|
|
|
|
return None
|
|
|
|
|
2020-05-21 22:48:00 +00:00
|
|
|
def is_known(self, repository_id):
|
2020-02-14 02:52:51 +00:00
|
|
|
"""Return a bool if the repository is known."""
|
2020-05-21 22:48:00 +00:00
|
|
|
return str(repository_id) in [str(x.data.id) for x in self.repositories]
|
2020-02-14 02:52:51 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def sorted_by_name(self):
|
|
|
|
"""Return a sorted(by name) list of repository objects."""
|
|
|
|
return sorted(self.repositories, key=lambda x: x.display_name)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def sorted_by_repository_name(self):
|
|
|
|
"""Return a sorted(by repository_name) list of repository objects."""
|
2020-04-10 01:29:27 +00:00
|
|
|
return sorted(self.repositories, key=lambda x: x.data.full_name)
|
2020-02-14 02:52:51 +00:00
|
|
|
|
|
|
|
async def register_repository(self, full_name, category, check=True):
|
|
|
|
"""Register a repository."""
|
2020-04-10 01:29:27 +00:00
|
|
|
await register_repository(full_name, category, check=True)
|
2020-02-14 02:52:51 +00:00
|
|
|
|
|
|
|
async def startup_tasks(self):
|
|
|
|
"""Tasks tha are started after startup."""
|
|
|
|
self.system.status.background_task = True
|
2020-04-10 01:29:27 +00:00
|
|
|
await self.hass.async_add_executor_job(setup_extra_stores)
|
2020-02-14 02:52:51 +00:00
|
|
|
self.hass.bus.async_fire("hacs/status", {})
|
|
|
|
|
|
|
|
await self.handle_critical_repositories_startup()
|
|
|
|
await self.handle_critical_repositories()
|
|
|
|
await self.load_known_repositories()
|
2020-04-10 01:29:27 +00:00
|
|
|
await self.clear_out_removed_repositories()
|
2020-02-14 02:52:51 +00:00
|
|
|
|
|
|
|
self.recuring_tasks.append(
|
|
|
|
async_track_time_interval(
|
|
|
|
self.hass, self.recuring_tasks_installed, timedelta(minutes=30)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
self.recuring_tasks.append(
|
|
|
|
async_track_time_interval(
|
|
|
|
self.hass, self.recuring_tasks_all, timedelta(minutes=800)
|
|
|
|
)
|
|
|
|
)
|
2020-05-03 20:23:17 +00:00
|
|
|
self.recuring_tasks.append(
|
|
|
|
async_track_time_interval(
|
|
|
|
self.hass, self.prosess_queue, timedelta(minutes=10)
|
|
|
|
)
|
|
|
|
)
|
2020-02-14 02:52:51 +00:00
|
|
|
|
|
|
|
self.hass.bus.async_fire("hacs/reload", {"force": True})
|
|
|
|
await self.recuring_tasks_installed()
|
|
|
|
|
2020-05-03 20:23:17 +00:00
|
|
|
await self.prosess_queue()
|
|
|
|
|
2020-02-14 02:52:51 +00:00
|
|
|
self.system.status.startup = False
|
|
|
|
self.system.status.background_task = False
|
|
|
|
self.hass.bus.async_fire("hacs/status", {})
|
|
|
|
await self.data.async_write()
|
|
|
|
|
|
|
|
async def handle_critical_repositories_startup(self):
|
|
|
|
"""Handled critical repositories during startup."""
|
|
|
|
alert = False
|
|
|
|
critical = await async_load_from_store(self.hass, "critical")
|
|
|
|
if not critical:
|
|
|
|
return
|
|
|
|
for repo in critical:
|
|
|
|
if not repo["acknowledged"]:
|
|
|
|
alert = True
|
|
|
|
if alert:
|
|
|
|
self.logger.critical("URGENT!: Check the HACS panel!")
|
|
|
|
self.hass.components.persistent_notification.create(
|
|
|
|
title="URGENT!", message="**Check the HACS panel!**"
|
|
|
|
)
|
|
|
|
|
|
|
|
async def handle_critical_repositories(self):
|
|
|
|
"""Handled critical repositories during runtime."""
|
|
|
|
# Get critical repositories
|
|
|
|
instored = []
|
|
|
|
critical = []
|
|
|
|
was_installed = False
|
|
|
|
|
|
|
|
try:
|
|
|
|
critical = await self.data_repo.get_contents("critical")
|
|
|
|
critical = json.loads(critical.content)
|
2020-05-21 22:48:00 +00:00
|
|
|
except AIOGitHubAPIException:
|
2020-02-14 02:52:51 +00:00
|
|
|
pass
|
|
|
|
|
|
|
|
if not critical:
|
|
|
|
self.logger.debug("No critical repositories")
|
|
|
|
return
|
|
|
|
|
|
|
|
stored_critical = await async_load_from_store(self.hass, "critical")
|
|
|
|
|
|
|
|
for stored in stored_critical or []:
|
|
|
|
instored.append(stored["repository"])
|
|
|
|
|
|
|
|
stored_critical = []
|
|
|
|
|
|
|
|
for repository in critical:
|
2020-04-10 01:29:27 +00:00
|
|
|
removed_repo = get_removed(repository["repository"])
|
|
|
|
removed_repo.removal_type = "critical"
|
2020-02-14 02:52:51 +00:00
|
|
|
repo = self.get_by_name(repository["repository"])
|
|
|
|
|
|
|
|
stored = {
|
|
|
|
"repository": repository["repository"],
|
|
|
|
"reason": repository["reason"],
|
|
|
|
"link": repository["link"],
|
|
|
|
"acknowledged": True,
|
|
|
|
}
|
|
|
|
if repository["repository"] not in instored:
|
|
|
|
if repo is not None and repo.installed:
|
|
|
|
self.logger.critical(
|
|
|
|
f"Removing repository {repository['repository']}, it is marked as critical"
|
|
|
|
)
|
|
|
|
was_installed = True
|
|
|
|
stored["acknowledged"] = False
|
|
|
|
# Uninstall from HACS
|
|
|
|
repo.remove()
|
|
|
|
await repo.uninstall()
|
|
|
|
stored_critical.append(stored)
|
2020-04-10 01:29:27 +00:00
|
|
|
removed_repo.update_data(stored)
|
2020-02-14 02:52:51 +00:00
|
|
|
|
|
|
|
# Save to FS
|
|
|
|
await async_save_to_store(self.hass, "critical", stored_critical)
|
|
|
|
|
|
|
|
# Resart HASS
|
|
|
|
if was_installed:
|
|
|
|
self.logger.critical("Resarting Home Assistant")
|
|
|
|
self.hass.async_create_task(self.hass.async_stop(100))
|
|
|
|
|
2020-05-03 20:23:17 +00:00
|
|
|
async def prosess_queue(self, notarealarg=None):
|
|
|
|
"""Recuring tasks for installed repositories."""
|
|
|
|
if not self.queue.has_pending_tasks:
|
|
|
|
self.logger.debug("Nothing in the queue")
|
|
|
|
return
|
|
|
|
if self.queue.running:
|
|
|
|
self.logger.debug("Queue is already running")
|
|
|
|
return
|
|
|
|
|
|
|
|
can_update = await get_fetch_updates_for(self.github)
|
|
|
|
if can_update == 0:
|
|
|
|
self.logger.info(
|
|
|
|
"HACS is ratelimited, repository updates will resume later."
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
self.system.status.background_task = True
|
|
|
|
self.hass.bus.async_fire("hacs/status", {})
|
|
|
|
await self.queue.execute(can_update)
|
|
|
|
self.system.status.background_task = False
|
|
|
|
self.hass.bus.async_fire("hacs/status", {})
|
|
|
|
|
2020-02-14 02:52:51 +00:00
|
|
|
async def recuring_tasks_installed(self, notarealarg=None):
|
|
|
|
"""Recuring tasks for installed repositories."""
|
|
|
|
self.logger.debug(
|
|
|
|
"Starting recuring background task for installed repositories"
|
|
|
|
)
|
|
|
|
self.system.status.background_task = True
|
|
|
|
self.hass.bus.async_fire("hacs/status", {})
|
2020-05-21 22:48:00 +00:00
|
|
|
|
2020-02-14 02:52:51 +00:00
|
|
|
for repository in self.repositories:
|
|
|
|
if (
|
2020-05-21 22:48:00 +00:00
|
|
|
repository.data.installed
|
2020-04-10 01:29:27 +00:00
|
|
|
and repository.data.category in self.common.categories
|
2020-02-14 02:52:51 +00:00
|
|
|
):
|
2020-05-03 20:23:17 +00:00
|
|
|
self.queue.add(self.factory.safe_update(repository))
|
2020-02-14 02:52:51 +00:00
|
|
|
|
|
|
|
await self.handle_critical_repositories()
|
|
|
|
self.system.status.background_task = False
|
|
|
|
self.hass.bus.async_fire("hacs/status", {})
|
|
|
|
await self.data.async_write()
|
|
|
|
self.logger.debug("Recuring background task for installed repositories done")
|
|
|
|
|
|
|
|
async def recuring_tasks_all(self, notarealarg=None):
|
|
|
|
"""Recuring tasks for all repositories."""
|
|
|
|
self.logger.debug("Starting recuring background task for all repositories")
|
2020-04-10 01:29:27 +00:00
|
|
|
await self.hass.async_add_executor_job(setup_extra_stores)
|
2020-02-14 02:52:51 +00:00
|
|
|
self.system.status.background_task = True
|
|
|
|
self.hass.bus.async_fire("hacs/status", {})
|
2020-05-21 22:48:00 +00:00
|
|
|
|
2020-02-14 02:52:51 +00:00
|
|
|
for repository in self.repositories:
|
2020-04-10 01:29:27 +00:00
|
|
|
if repository.data.category in self.common.categories:
|
2020-05-03 20:23:17 +00:00
|
|
|
self.queue.add(self.factory.safe_common_update(repository))
|
2020-02-14 02:52:51 +00:00
|
|
|
|
|
|
|
await self.load_known_repositories()
|
2020-04-10 01:29:27 +00:00
|
|
|
await self.clear_out_removed_repositories()
|
2020-02-14 02:52:51 +00:00
|
|
|
self.system.status.background_task = False
|
|
|
|
await self.data.async_write()
|
|
|
|
self.hass.bus.async_fire("hacs/status", {})
|
|
|
|
self.hass.bus.async_fire("hacs/repository", {"action": "reload"})
|
|
|
|
self.logger.debug("Recuring background task for all repositories done")
|
|
|
|
|
2020-04-10 01:29:27 +00:00
|
|
|
async def clear_out_removed_repositories(self):
|
2020-02-14 02:52:51 +00:00
|
|
|
"""Clear out blaclisted repositories."""
|
|
|
|
need_to_save = False
|
2020-04-10 01:29:27 +00:00
|
|
|
for removed in removed_repositories:
|
2020-05-21 22:48:00 +00:00
|
|
|
repository = self.get_by_name(removed.repository)
|
|
|
|
if repository is not None:
|
|
|
|
if repository.data.installed and removed.removal_type != "critical":
|
2020-02-14 02:52:51 +00:00
|
|
|
self.logger.warning(
|
2020-04-10 01:29:27 +00:00
|
|
|
f"You have {repository.data.full_name} installed with HACS "
|
|
|
|
+ f"this repository has been removed, please consider removing it. "
|
|
|
|
+ f"Removal reason ({removed.removal_type})"
|
2020-02-14 02:52:51 +00:00
|
|
|
)
|
|
|
|
else:
|
|
|
|
need_to_save = True
|
|
|
|
repository.remove()
|
|
|
|
|
|
|
|
if need_to_save:
|
|
|
|
await self.data.async_write()
|
|
|
|
|
|
|
|
async def get_repositories(self):
|
|
|
|
"""Return a list of repositories."""
|
|
|
|
repositories = {}
|
|
|
|
for category in self.common.categories:
|
|
|
|
repositories[category] = await get_default_repos_lists(
|
2020-04-10 01:29:27 +00:00
|
|
|
self.session, self.configuration.token, category
|
2020-02-14 02:52:51 +00:00
|
|
|
)
|
|
|
|
org = await get_default_repos_orgs(self.github, category)
|
|
|
|
for repo in org:
|
|
|
|
repositories[category].append(repo)
|
|
|
|
return repositories
|
|
|
|
|
|
|
|
async def load_known_repositories(self):
|
|
|
|
"""Load known repositories."""
|
|
|
|
self.logger.info("Loading known repositories")
|
|
|
|
repositories = await self.get_repositories()
|
|
|
|
|
2020-04-10 01:29:27 +00:00
|
|
|
for item in await get_default_repos_lists(
|
|
|
|
self.session, self.configuration.token, "removed"
|
|
|
|
):
|
|
|
|
removed = get_removed(item["repository"])
|
|
|
|
removed.reason = item.get("reason")
|
|
|
|
removed.link = item.get("link")
|
|
|
|
removed.removal_type = item.get("removal_type")
|
2020-02-14 02:52:51 +00:00
|
|
|
|
|
|
|
for category in repositories:
|
|
|
|
for repo in repositories[category]:
|
2020-04-10 01:29:27 +00:00
|
|
|
if is_removed(repo):
|
2020-02-14 02:52:51 +00:00
|
|
|
continue
|
2020-05-21 22:48:00 +00:00
|
|
|
repository = self.get_by_name(repo)
|
|
|
|
if repository is not None:
|
|
|
|
if str(repository.data.id) not in self.common.default:
|
|
|
|
self.common.default.append(str(repository.data.id))
|
|
|
|
else:
|
|
|
|
continue
|
2020-02-14 02:52:51 +00:00
|
|
|
continue
|
2020-05-03 20:23:17 +00:00
|
|
|
self.queue.add(self.factory.safe_register(repo, category))
|