def test_data_structure(): RepositoryData.create_from_dict({"pushed_at": ""}) RepositoryData.create_from_dict({"country": ""}) RepositoryData.create_from_dict({"country": [""]}) RepositoryData.create_from_dict({"pushed_at": "1970-01-01T00:00:00"}) data = RepositoryData.create_from_dict(repository_data) assert data.stars == 999 assert isinstance(data.to_json(), dict)
def test_data_update(): data = RepositoryData.create_from_dict({}) assert not data.fork data.update_data({"fork": True}) data.update_data({"pushed_at": "1970-01-01T00:00:00"}) data.update_data({"country": ""}) data.update_data({"country": [""]}) data.update_data({"pushed_at": ""}) assert data.fork
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
class MockRepo(RepositoryMethodInstall, RepositoryMethodPreInstall, RepositoryMethodPostInstall): logger = getLogger() content = MockContent() validate = Validate() can_install = False data = RepositoryData() tree = [] hacs = get_hacs() async def update_repository(self): pass
class HacsRepository(RepositoryHelpers): """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 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("::error:: 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 download_zip_files(self, validate): """Download ZIP archive from repository release.""" download_queue = QueueManager() 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 []: download_queue.add( self.async_download_zip_file(content, validate)) await download_queue.execute() except (Exception, BaseException): validate.errors.append(f"Download was not completed") return validate async def async_download_zip_file(self, content, validate): """Download ZIP archive from repository release.""" try: filecontent = await async_download_file(content.download_url) if filecontent is None: validate.errors.append(f"[{content.name}] was not downloaded") return 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} completed") return validate.errors.append(f"[{content.name}] was not downloaded") except (Exception, BaseException): validate.errors.append(f"Download was not completed") return validate async def download_content(self, validate, _directory_path, _local_directory, _ref): """Download the content of a directory.""" from custom_components.hacs.helpers.functions.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( "::error:: 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"::error:: 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, BaseException): # 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 = f"{self.content.path.local}/{self.data.name}.py" 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): if not is_safe_to_remove(local_path): return False 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, BaseException) as exception: self.logger.debug(f"Removing {local_path} failed with {exception}") return False return True