"""Class representing Bitwarden service""" import base64 import subprocess from typing import List from os import path from pathlib import Path # from enum import Enum from selfprivacy_api.jobs import Job from selfprivacy_api.services.service import Service, ServiceStatus from selfprivacy_api.utils.block_devices import BlockDevice from selfprivacy_api.services.test_service.icon import BITWARDEN_ICON DEFAULT_DELAY = 0 class DummyService(Service): """A test service""" folders: List[str] = [] startstop_delay = 0.0 backuppable = True movable = True fail_to_stop = False # if False, we try to actually move simulate_moving = True drive = "sda1" def __init_subclass__(cls, folders: List[str]): cls.folders = folders def __init__(self): # Maybe init it with some dummy files right here # currently it is done in a fixture but if we do it here # then we can do some convenience methods of writing and reading # from test files so that # we can easily check integrity in numerous restore tests super().__init__() with open(self.status_file(), "w") as file: file.write(ServiceStatus.ACTIVE.value) @staticmethod def get_id() -> str: """Return service id.""" return "testservice" @staticmethod def get_display_name() -> str: """Return service display name.""" return "Test Service" @staticmethod def get_description() -> str: """Return service description.""" return "A small service used for test purposes. Does nothing." @staticmethod def get_svg_icon() -> str: """Read SVG icon from file and return it as base64 encoded string.""" # return "" return base64.b64encode(BITWARDEN_ICON.encode("utf-8")).decode("utf-8") @classmethod def is_movable(cls) -> bool: return cls.movable @staticmethod def is_required() -> bool: return False @staticmethod def get_backup_description() -> str: return "How did we get here?" @classmethod def status_file(cls) -> str: dir = cls.folders[0] # We do not want to store our state in our declared folders # Because they are moved and tossed in tests wildly parent = Path(dir).parent return path.join(parent, "service_status") @classmethod def set_status(cls, status: ServiceStatus): with open(cls.status_file(), "w") as file: file.write(status.value) @classmethod def get_status(cls) -> ServiceStatus: filepath = cls.status_file() if filepath in [None, ""]: raise ValueError("We do not have a path for our test dummy status file!") if not path.exists(filepath): raise FileNotFoundError(filepath) with open(cls.status_file(), "r") as file: status_string = file.read().strip() if status_string in [None, ""]: raise NotImplementedError( f"It appears our test service no longer has any status in the statusfile. Filename = {cls.status_file}, status string inside is '{status_string}' (quoted) " ) return ServiceStatus[status_string] @classmethod def change_status_with_async_delay( cls, new_status: ServiceStatus, delay_sec: float ): """simulating a delay on systemd side""" if not isinstance(new_status, ServiceStatus): raise ValueError( f"received an invalid new status for test service. new status: {str(new_status)}" ) if delay_sec == 0: cls.set_status(new_status) return status_file = cls.status_file() command = [ "bash", "-c", f" sleep {delay_sec} && echo {new_status.value} > {status_file}", ] subprocess.Popen(command) @classmethod def set_backuppable(cls, new_value: bool) -> None: """For tests: because can_be_backed_up is static, we can only set it up dynamically for tests via a classmethod""" cls.backuppable = new_value @classmethod def set_movable(cls, new_value: bool) -> None: """For tests: because is_movale is static, we can only set it up dynamically for tests via a classmethod""" cls.movable = new_value @classmethod def can_be_backed_up(cls) -> bool: """`True` if the service can be backed up.""" return cls.backuppable @classmethod def set_delay(cls, new_delay_sec: float) -> None: cls.startstop_delay = new_delay_sec @classmethod def set_drive(cls, new_drive: str) -> None: cls.drive = new_drive @classmethod def set_simulated_moves(cls, enabled: bool) -> None: """If True, this service will not actually call moving code when moved""" cls.simulate_moving = enabled @classmethod def simulate_fail_to_stop(cls, value: bool): cls.fail_to_stop = value @classmethod def stop(cls): # simulate a failing service unable to stop if not cls.get_status() == ServiceStatus.FAILED: cls.set_status(ServiceStatus.DEACTIVATING) if cls.fail_to_stop: cls.change_status_with_async_delay( ServiceStatus.FAILED, cls.startstop_delay ) else: cls.change_status_with_async_delay( ServiceStatus.INACTIVE, cls.startstop_delay ) @classmethod def start(cls): cls.set_status(ServiceStatus.ACTIVATING) cls.change_status_with_async_delay(ServiceStatus.ACTIVE, cls.startstop_delay) @classmethod def restart(cls): cls.set_status(ServiceStatus.RELOADING) # is a correct one? cls.change_status_with_async_delay(ServiceStatus.ACTIVE, cls.startstop_delay) @classmethod def get_configuration(cls): return {} @classmethod def set_configuration(cls, config_items): return super().set_configuration(config_items) @staticmethod def get_storage_usage() -> int: storage_usage = 0 return storage_usage @classmethod def get_drive(cls) -> str: return cls.drive @classmethod def get_folders(cls) -> List[str]: return cls.folders def do_move_to_volume(self, volume: BlockDevice, job: Job) -> Job: if self.simulate_moving is False: return super(DummyService, self).do_move_to_volume(volume, job) else: self.set_drive(volume.name) return job