"""Repository.""" # pylint: disable=broad-except, bad-continuation, no-member import json import os import tempfile import zipfile from integrationhelper import Validate from aiogithubapi import AIOGitHubAPIException from .manifest import HacsManifest from ..helpers.misc import get_repository_name from ..handler.download import async_download_file, async_save_file from ..helpers.misc import version_left_higher_then_right from ..helpers.install import install_repository, version_to_install from custom_components.hacs.hacsbase.exceptions import HacsException from custom_components.hacs.store import async_remove_store from custom_components.hacs.globals import get_hacs from custom_components.hacs.helpers.information import ( get_info_md_content, get_repository, ) from custom_components.hacs.helpers.validate_repository import ( common_validate, common_update_data, ) from custom_components.hacs.repositories.repositorydata import RepositoryData class RepositoryVersions: """Versions.""" available = None available_commit = None installed = None installed_commit = None class RepositoryStatus: """Repository status.""" hide = False installed = False last_updated = None new = True selected_tag = None show_beta = False track = True updated_info = False first_install = True class RepositoryInformation: """RepositoryInformation.""" additional_info = None authors = [] category = None default_branch = None description = "" state = None full_name = None file_name = None javascript_type = None homeassistant_version = None last_updated = None uid = None stars = 0 info = None name = None topics = [] class RepositoryReleases: """RepositoyReleases.""" last_release = None last_release_object = None last_release_object_downloads = None published_tags = [] objects = [] releases = False downloads = None class RepositoryPath: """RepositoryPath.""" local = None remote = None class RepositoryContent: """RepositoryContent.""" path = None files = [] objects = [] single = False class HacsRepository: """HacsRepository.""" def __init__(self): """Set up HacsRepository.""" self.hacs = get_hacs() self.data = RepositoryData() self.content = RepositoryContent() self.content.path = RepositoryPath() self.information = RepositoryInformation() self.repository_object = None self.status = RepositoryStatus() self.state = None self.force_branch = False self.integration_manifest = {} self.repository_manifest = HacsManifest.from_dict({}) self.validate = Validate() self.releases = RepositoryReleases() self.versions = RepositoryVersions() self.pending_restart = False self.tree = [] self.treefiles = [] self.ref = None @property def pending_upgrade(self): """Return pending upgrade.""" if not self.can_install: return False if self.data.installed: if self.data.selected_tag is not None: if self.data.selected_tag == self.data.default_branch: if self.data.installed_commit != self.data.last_commit: return True return False if self.display_installed_version != self.display_available_version: return True return False @property def custom(self): """Return flag if the repository is custom.""" if self.data.full_name.split("/")[0] in ["custom-components", "custom-cards"]: return False if str(self.data.id) in [str(x) for x in self.hacs.common.default]: return False if self.data.full_name == "hacs/integration": return False return True @property def can_install(self): """Return bool if repository can be installed.""" target = None if self.data.homeassistant is not None: target = self.data.homeassistant if self.data.homeassistant is not None: target = self.data.homeassistant if target is not None: if self.data.releases: if not version_left_higher_then_right( self.hacs.system.ha_version, target ): return False return True @property def display_name(self): """Return display name.""" return get_repository_name(self) @property def display_status(self): """Return display_status.""" if self.data.new: status = "new" elif self.pending_restart: status = "pending-restart" elif self.pending_upgrade: status = "pending-upgrade" elif self.data.installed: status = "installed" else: status = "default" return status @property def display_status_description(self): """Return display_status_description.""" description = { "default": "Not installed.", "pending-restart": "Restart pending.", "pending-upgrade": "Upgrade pending.", "installed": "No action required.", "new": "This is a newly added repository.", } return description[self.display_status] @property def display_installed_version(self): """Return display_authors""" if self.data.installed_version is not None: installed = self.data.installed_version else: if self.data.installed_commit is not None: installed = self.data.installed_commit else: installed = "" return installed @property def display_available_version(self): """Return display_authors""" if self.data.last_version is not None: available = self.data.last_version else: if self.data.last_commit is not None: available = self.data.last_commit else: available = "" return available @property def display_version_or_commit(self): """Does the repositoriy use releases or commits?""" if self.data.releases: version_or_commit = "version" else: version_or_commit = "commit" return version_or_commit @property def main_action(self): """Return the main action.""" actions = { "new": "INSTALL", "default": "INSTALL", "installed": "REINSTALL", "pending-restart": "REINSTALL", "pending-upgrade": "UPGRADE", } return actions[self.display_status] async def common_validate(self, ignore_issues=False): """Common validation steps of the repository.""" await common_validate(self, ignore_issues) async def common_registration(self): """Common registration steps of the repository.""" # Attach repository if self.repository_object is None: self.repository_object = await get_repository( self.hacs.session, self.hacs.configuration.token, self.data.full_name ) self.data.update_data(self.repository_object.attributes) # Set topics self.data.topics = self.data.topics # Set stargazers_count self.data.stargazers_count = self.data.stargazers_count # Set description self.data.description = self.data.description if self.hacs.action: if self.data.description is None or len(self.data.description) == 0: raise HacsException("Missing repository description") async def common_update(self, ignore_issues=False): """Common information update steps of the repository.""" self.logger.debug("Getting repository information") # Attach repository await common_update_data(self, ignore_issues) # Update last updaeted self.data.last_updated = self.repository_object.attributes.get("pushed_at", 0) # Update last available commit await self.repository_object.set_last_commit() self.data.last_commit = self.repository_object.last_commit # Get the content of hacs.json await self.get_repository_manifest_content() # Update "info.md" self.information.additional_info = await get_info_md_content(self) async def install(self): """Common installation steps of the repository.""" await install_repository(self) async def download_zip(self, validate): """Download ZIP archive from repository release.""" try: contents = False for release in self.releases.objects: self.logger.info(f"ref: {self.ref} --- tag: {release.tag_name}") if release.tag_name == self.ref.split("/")[1]: contents = release.assets if not contents: return validate for content in contents or []: filecontent = await async_download_file(content.download_url) if filecontent is None: validate.errors.append(f"[{content.name}] was not downloaded.") continue result = await async_save_file( f"{tempfile.gettempdir()}/{self.data.filename}", filecontent ) with zipfile.ZipFile( f"{tempfile.gettempdir()}/{self.data.filename}", "r" ) as zip_file: zip_file.extractall(self.content.path.local) if result: self.logger.info(f"download of {content.name} complete") continue validate.errors.append(f"[{content.name}] was not downloaded.") except Exception: validate.errors.append(f"Download was not complete.") return validate async def download_content(self, validate, directory_path, local_directory, ref): """Download the content of a directory.""" from custom_components.hacs.helpers.download import download_content validate = await download_content(self) return validate async def get_repository_manifest_content(self): """Get the content of the hacs.json file.""" if not "hacs.json" in [x.filename for x in self.tree]: if self.hacs.action: raise HacsException("No hacs.json file in the root of the repository.") return if self.hacs.action: self.logger.info("Found hacs.json") self.ref = version_to_install(self) try: manifest = await self.repository_object.get_contents("hacs.json", self.ref) self.repository_manifest = HacsManifest.from_dict( json.loads(manifest.content) ) self.data.update_data(json.loads(manifest.content)) except (AIOGitHubAPIException, Exception) as exception: # Gotta Catch 'Em All if self.hacs.action: raise HacsException(f"hacs.json file is not valid ({exception}).") if self.hacs.action: self.logger.info("hacs.json is valid") def remove(self): """Run remove tasks.""" self.logger.info("Starting removal") if self.data.id in self.hacs.common.installed: self.hacs.common.installed.remove(self.data.id) for repository in self.hacs.repositories: if repository.data.id == self.data.id: self.hacs.repositories.remove(repository) async def uninstall(self): """Run uninstall tasks.""" self.logger.info("Uninstalling") if not await self.remove_local_directory(): raise HacsException("Could not uninstall") self.data.installed = False if self.data.category == "integration": if self.data.config_flow: await self.reload_custom_components() else: self.pending_restart = True elif self.data.category == "theme": try: await self.hacs.hass.services.async_call( "frontend", "reload_themes", {} ) except Exception: # pylint: disable=broad-except pass if self.data.full_name in self.hacs.common.installed: self.hacs.common.installed.remove(self.data.full_name) await async_remove_store(self.hacs.hass, f"hacs/{self.data.id}.hacs") self.data.installed_version = None self.data.installed_commit = None self.hacs.hass.bus.async_fire( "hacs/repository", {"id": 1337, "action": "uninstall", "repository": self.data.full_name}, ) async def remove_local_directory(self): """Check the local directory.""" import shutil from asyncio import sleep try: if self.data.category == "python_script": local_path = "{}/{}.py".format(self.content.path.local, self.data.name) elif self.data.category == "theme": if os.path.exists( f"{self.hacs.system.config_path}/{self.hacs.configuration.theme_path}/{self.data.name}.yaml" ): os.remove( f"{self.hacs.system.config_path}/{self.hacs.configuration.theme_path}/{self.data.name}.yaml" ) local_path = self.content.path.local elif self.data.category == "integration": if not self.data.domain: self.logger.error("Missing domain") return False local_path = self.content.path.local else: local_path = self.content.path.local if os.path.exists(local_path): self.logger.debug(f"Removing {local_path}") if self.data.category in ["python_script"]: os.remove(local_path) else: shutil.rmtree(local_path) while os.path.exists(local_path): await sleep(1) except Exception as exception: self.logger.debug(f"Removing {local_path} failed with {exception}") return False return True