mirror of
https://git.selfprivacy.org/SelfPrivacy/selfprivacy-rest-api.git
synced 2025-01-23 09:16:51 +00:00
refactor(backups): fix typing errors
This commit is contained in:
parent
1e840f8cff
commit
f27a3df807
|
@ -1,3 +1,4 @@
|
||||||
|
from operator import add
|
||||||
from typing import List, Optional
|
from typing import List, Optional
|
||||||
from datetime import datetime, timedelta
|
from datetime import datetime, timedelta
|
||||||
from os import statvfs
|
from os import statvfs
|
||||||
|
@ -9,7 +10,9 @@ from selfprivacy_api.utils import ReadUserData, WriteUserData
|
||||||
from selfprivacy_api.services import get_service_by_id
|
from selfprivacy_api.services import get_service_by_id
|
||||||
from selfprivacy_api.services.service import Service
|
from selfprivacy_api.services.service import Service
|
||||||
|
|
||||||
from selfprivacy_api.graphql.queries.providers import BackupProvider
|
from selfprivacy_api.graphql.queries.providers import (
|
||||||
|
BackupProvider as BackupProviderEnum,
|
||||||
|
)
|
||||||
|
|
||||||
from selfprivacy_api.backup.providers.provider import AbstractBackupProvider
|
from selfprivacy_api.backup.providers.provider import AbstractBackupProvider
|
||||||
from selfprivacy_api.backup.providers import get_provider
|
from selfprivacy_api.backup.providers import get_provider
|
||||||
|
@ -33,12 +36,15 @@ DEFAULT_JSON_PROVIDER = {
|
||||||
class Backups:
|
class Backups:
|
||||||
"""A singleton controller for backups"""
|
"""A singleton controller for backups"""
|
||||||
|
|
||||||
provider: AbstractBackupProvider
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def set_localfile_repo(file_path: str):
|
def set_localfile_repo(file_path: str):
|
||||||
ProviderClass = get_provider(BackupProvider.FILE)
|
ProviderClass = get_provider(BackupProviderEnum.FILE)
|
||||||
provider = ProviderClass(login="", key="", location=file_path, repo_id="")
|
provider = ProviderClass(
|
||||||
|
login="",
|
||||||
|
key="",
|
||||||
|
location=file_path,
|
||||||
|
repo_id="",
|
||||||
|
)
|
||||||
Storage.store_provider(provider)
|
Storage.store_provider(provider)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
|
@ -67,7 +73,14 @@ class Backups:
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _service_ids_to_back_up(time: datetime) -> List[str]:
|
def _service_ids_to_back_up(time: datetime) -> List[str]:
|
||||||
services = Storage.services_with_autobackup()
|
services = Storage.services_with_autobackup()
|
||||||
return [id for id in services if Backups.is_time_to_backup_service(id, time)]
|
return [
|
||||||
|
id
|
||||||
|
for id in services
|
||||||
|
if Backups.is_time_to_backup_service(
|
||||||
|
id,
|
||||||
|
time,
|
||||||
|
)
|
||||||
|
]
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def services_to_back_up(time: datetime) -> List[Service]:
|
def services_to_back_up(time: datetime) -> List[Service]:
|
||||||
|
@ -75,14 +88,17 @@ class Backups:
|
||||||
for id in Backups._service_ids_to_back_up(time):
|
for id in Backups._service_ids_to_back_up(time):
|
||||||
service = get_service_by_id(id)
|
service = get_service_by_id(id)
|
||||||
if service is None:
|
if service is None:
|
||||||
raise ValueError("Cannot look up a service scheduled for backup!")
|
raise ValueError(
|
||||||
|
"Cannot look up a service scheduled for backup!",
|
||||||
|
)
|
||||||
result.append(service)
|
result.append(service)
|
||||||
return result
|
return result
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def is_time_to_backup(time: datetime) -> bool:
|
def is_time_to_backup(time: datetime) -> bool:
|
||||||
"""
|
"""
|
||||||
Intended as a time validator for huey cron scheduler of automatic backups
|
Intended as a time validator for huey cron scheduler
|
||||||
|
of automatic backups
|
||||||
"""
|
"""
|
||||||
|
|
||||||
return Backups._service_ids_to_back_up(time) != []
|
return Backups._service_ids_to_back_up(time) != []
|
||||||
|
@ -97,7 +113,8 @@ class Backups:
|
||||||
|
|
||||||
last_backup = Storage.get_last_backup_time(service_id)
|
last_backup = Storage.get_last_backup_time(service_id)
|
||||||
if last_backup is None:
|
if last_backup is None:
|
||||||
return True # queue a backup immediately if there are no previous backups
|
# queue a backup immediately if there are no previous backups
|
||||||
|
return True
|
||||||
|
|
||||||
if time > last_backup + timedelta(minutes=period):
|
if time > last_backup + timedelta(minutes=period):
|
||||||
return True
|
return True
|
||||||
|
@ -121,7 +138,8 @@ class Backups:
|
||||||
def set_autobackup_period_minutes(minutes: int):
|
def set_autobackup_period_minutes(minutes: int):
|
||||||
"""
|
"""
|
||||||
0 and negative numbers are equivalent to disable.
|
0 and negative numbers are equivalent to disable.
|
||||||
Setting to a positive number may result in a backup very soon if some services are not backed up.
|
Setting to a positive number may result in a backup very soon
|
||||||
|
if some services are not backed up.
|
||||||
"""
|
"""
|
||||||
if minutes <= 0:
|
if minutes <= 0:
|
||||||
Backups.disable_all_autobackup()
|
Backups.disable_all_autobackup()
|
||||||
|
@ -130,7 +148,10 @@ class Backups:
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def disable_all_autobackup():
|
def disable_all_autobackup():
|
||||||
"""disables all automatic backing up, but does not change per-service settings"""
|
"""
|
||||||
|
Disables all automatic backing up,
|
||||||
|
but does not change per-service settings
|
||||||
|
"""
|
||||||
Storage.delete_backup_period()
|
Storage.delete_backup_period()
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
|
@ -138,17 +159,38 @@ class Backups:
|
||||||
return Backups.lookup_provider()
|
return Backups.lookup_provider()
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def set_provider(kind: str, login: str, key: str, location: str, repo_id: str = ""):
|
def set_provider(
|
||||||
provider = Backups.construct_provider(kind, login, key, location, repo_id)
|
kind: BackupProviderEnum,
|
||||||
|
login: str,
|
||||||
|
key: str,
|
||||||
|
location: str,
|
||||||
|
repo_id: str = "",
|
||||||
|
):
|
||||||
|
provider = Backups.construct_provider(
|
||||||
|
kind,
|
||||||
|
login,
|
||||||
|
key,
|
||||||
|
location,
|
||||||
|
repo_id,
|
||||||
|
)
|
||||||
Storage.store_provider(provider)
|
Storage.store_provider(provider)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def construct_provider(
|
def construct_provider(
|
||||||
kind: str, login: str, key: str, location: str, repo_id: str = ""
|
kind: BackupProviderEnum,
|
||||||
):
|
login: str,
|
||||||
provider_class = get_provider(BackupProvider[kind])
|
key: str,
|
||||||
|
location: str,
|
||||||
|
repo_id: str = "",
|
||||||
|
) -> AbstractBackupProvider:
|
||||||
|
provider_class = get_provider(kind)
|
||||||
|
|
||||||
return provider_class(login=login, key=key, location=location, repo_id=repo_id)
|
return provider_class(
|
||||||
|
login=login,
|
||||||
|
key=key,
|
||||||
|
location=location,
|
||||||
|
repo_id=repo_id,
|
||||||
|
)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def reset(reset_json=True):
|
def reset(reset_json=True):
|
||||||
|
@ -156,7 +198,8 @@ class Backups:
|
||||||
if reset_json:
|
if reset_json:
|
||||||
try:
|
try:
|
||||||
Backups.reset_provider_json()
|
Backups.reset_provider_json()
|
||||||
except FileNotFoundError: # if there is no userdata file, we do not need to reset it
|
except FileNotFoundError:
|
||||||
|
# if there is no userdata file, we do not need to reset it
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
|
@ -175,7 +218,7 @@ class Backups:
|
||||||
return json_provider
|
return json_provider
|
||||||
|
|
||||||
none_provider = Backups.construct_provider(
|
none_provider = Backups.construct_provider(
|
||||||
"NONE", login="", key="", location=""
|
BackupProviderEnum.NONE, login="", key="", location=""
|
||||||
)
|
)
|
||||||
Storage.store_provider(none_provider)
|
Storage.store_provider(none_provider)
|
||||||
return none_provider
|
return none_provider
|
||||||
|
@ -200,15 +243,18 @@ class Backups:
|
||||||
|
|
||||||
if provider_dict == DEFAULT_JSON_PROVIDER:
|
if provider_dict == DEFAULT_JSON_PROVIDER:
|
||||||
return None
|
return None
|
||||||
|
try:
|
||||||
return Backups.construct_provider(
|
return Backups.construct_provider(
|
||||||
kind=provider_dict["provider"],
|
kind=BackupProviderEnum[provider_dict["provider"]],
|
||||||
login=provider_dict["accountId"],
|
login=provider_dict["accountId"],
|
||||||
key=provider_dict["accountKey"],
|
key=provider_dict["accountKey"],
|
||||||
location=provider_dict["bucket"],
|
location=provider_dict["bucket"],
|
||||||
)
|
)
|
||||||
|
except KeyError:
|
||||||
|
return None
|
||||||
|
|
||||||
def reset_provider_json() -> AbstractBackupProvider:
|
@staticmethod
|
||||||
|
def reset_provider_json() -> None:
|
||||||
with WriteUserData() as user_data:
|
with WriteUserData() as user_data:
|
||||||
if "backblaze" in user_data.keys():
|
if "backblaze" in user_data.keys():
|
||||||
del user_data["backblaze"]
|
del user_data["backblaze"]
|
||||||
|
@ -216,12 +262,12 @@ class Backups:
|
||||||
user_data["backup"] = DEFAULT_JSON_PROVIDER
|
user_data["backup"] = DEFAULT_JSON_PROVIDER
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def load_provider_redis() -> AbstractBackupProvider:
|
def load_provider_redis() -> Optional[AbstractBackupProvider]:
|
||||||
provider_model = Storage.load_provider()
|
provider_model = Storage.load_provider()
|
||||||
if provider_model is None:
|
if provider_model is None:
|
||||||
return None
|
return None
|
||||||
return Backups.construct_provider(
|
return Backups.construct_provider(
|
||||||
provider_model.kind,
|
BackupProviderEnum[provider_model.kind],
|
||||||
provider_model.login,
|
provider_model.login,
|
||||||
provider_model.key,
|
provider_model.key,
|
||||||
provider_model.location,
|
provider_model.location,
|
||||||
|
@ -232,7 +278,7 @@ class Backups:
|
||||||
def back_up(service: Service):
|
def back_up(service: Service):
|
||||||
"""The top-level function to back up a service"""
|
"""The top-level function to back up a service"""
|
||||||
folders = service.get_folders()
|
folders = service.get_folders()
|
||||||
repo_name = service.get_id()
|
tag = service.get_id()
|
||||||
|
|
||||||
job = get_backup_job(service)
|
job = get_backup_job(service)
|
||||||
if job is None:
|
if job is None:
|
||||||
|
@ -241,8 +287,11 @@ class Backups:
|
||||||
|
|
||||||
try:
|
try:
|
||||||
service.pre_backup()
|
service.pre_backup()
|
||||||
snapshot = Backups.provider().backuper.start_backup(folders, repo_name)
|
snapshot = Backups.provider().backuper.start_backup(
|
||||||
Backups._store_last_snapshot(repo_name, snapshot)
|
folders,
|
||||||
|
tag,
|
||||||
|
)
|
||||||
|
Backups._store_last_snapshot(tag, snapshot)
|
||||||
service.post_restore()
|
service.post_restore()
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
Jobs.update(job, status=JobStatus.ERROR)
|
Jobs.update(job, status=JobStatus.ERROR)
|
||||||
|
@ -252,10 +301,7 @@ class Backups:
|
||||||
return snapshot
|
return snapshot
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def init_repo(service: Optional[Service] = None):
|
def init_repo():
|
||||||
if service is not None:
|
|
||||||
repo_name = service.get_id()
|
|
||||||
|
|
||||||
Backups.provider().backuper.init()
|
Backups.provider().backuper.init()
|
||||||
Storage.mark_as_init()
|
Storage.mark_as_init()
|
||||||
|
|
||||||
|
@ -274,7 +320,13 @@ class Backups:
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def get_snapshots(service: Service) -> List[Snapshot]:
|
def get_snapshots(service: Service) -> List[Snapshot]:
|
||||||
snapshots = Backups.get_all_snapshots()
|
snapshots = Backups.get_all_snapshots()
|
||||||
return [snap for snap in snapshots if snap.service_name == service.get_id()]
|
service_id = service.get_id()
|
||||||
|
return list(
|
||||||
|
filter(
|
||||||
|
lambda snap: snap.service_name == service_id,
|
||||||
|
snapshots,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def get_all_snapshots() -> List[Snapshot]:
|
def get_all_snapshots() -> List[Snapshot]:
|
||||||
|
@ -314,10 +366,12 @@ class Backups:
|
||||||
# to be deprecated/internalized in favor of restore_snapshot()
|
# to be deprecated/internalized in favor of restore_snapshot()
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def restore_service_from_snapshot(service: Service, snapshot_id: str):
|
def restore_service_from_snapshot(service: Service, snapshot_id: str):
|
||||||
repo_name = service.get_id()
|
|
||||||
folders = service.get_folders()
|
folders = service.get_folders()
|
||||||
|
|
||||||
Backups.provider().backuper.restore_from_backup(repo_name, snapshot_id, folders)
|
Backups.provider().backuper.restore_from_backup(
|
||||||
|
snapshot_id,
|
||||||
|
folders,
|
||||||
|
)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def assert_restorable(snapshot: Snapshot):
|
def assert_restorable(snapshot: Snapshot):
|
||||||
|
@ -327,45 +381,58 @@ class Backups:
|
||||||
f"snapshot has a nonexistent service: {snapshot.service_name}"
|
f"snapshot has a nonexistent service: {snapshot.service_name}"
|
||||||
)
|
)
|
||||||
|
|
||||||
needed_space = Backups.snapshot_restored_size(snapshot)
|
needed_space = Backups.service_snapshot_size(snapshot.id)
|
||||||
available_space = Backups.space_usable_for_service(service)
|
available_space = Backups.space_usable_for_service(service)
|
||||||
if needed_space > available_space:
|
if needed_space > available_space:
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
f"we only have {available_space} bytes but snapshot needs{ needed_space}"
|
f"we only have {available_space} bytes "
|
||||||
|
f"but snapshot needs {needed_space}"
|
||||||
)
|
)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def restore_snapshot(snapshot: Snapshot):
|
def restore_snapshot(snapshot: Snapshot):
|
||||||
service = get_service_by_id(snapshot.service_name)
|
service = get_service_by_id(snapshot.service_name)
|
||||||
|
|
||||||
|
if service is None:
|
||||||
|
raise ValueError(
|
||||||
|
f"snapshot has a nonexistent service: {snapshot.service_name}"
|
||||||
|
)
|
||||||
|
|
||||||
job = get_restore_job(service)
|
job = get_restore_job(service)
|
||||||
if job is None:
|
if job is None:
|
||||||
job = add_restore_job(snapshot)
|
job = add_restore_job(snapshot)
|
||||||
|
|
||||||
Jobs.update(job, status=JobStatus.RUNNING)
|
Jobs.update(
|
||||||
|
job,
|
||||||
|
status=JobStatus.RUNNING,
|
||||||
|
)
|
||||||
try:
|
try:
|
||||||
Backups.assert_restorable(snapshot)
|
Backups.assert_restorable(snapshot)
|
||||||
Backups.restore_service_from_snapshot(service, snapshot.id)
|
Backups.restore_service_from_snapshot(
|
||||||
|
service,
|
||||||
|
snapshot.id,
|
||||||
|
)
|
||||||
service.post_restore()
|
service.post_restore()
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
Jobs.update(job, status=JobStatus.ERROR)
|
Jobs.update(
|
||||||
|
job,
|
||||||
|
status=JobStatus.ERROR,
|
||||||
|
)
|
||||||
raise e
|
raise e
|
||||||
|
|
||||||
Jobs.update(job, status=JobStatus.FINISHED)
|
Jobs.update(
|
||||||
|
job,
|
||||||
@staticmethod
|
status=JobStatus.FINISHED,
|
||||||
def service_snapshot_size(service: Service, snapshot_id: str) -> float:
|
|
||||||
repo_name = service.get_id()
|
|
||||||
return Backups.provider().backuper.restored_size(repo_name, snapshot_id)
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def snapshot_restored_size(snapshot: Snapshot) -> float:
|
|
||||||
return Backups.service_snapshot_size(
|
|
||||||
get_service_by_id(snapshot.service_name), snapshot.id
|
|
||||||
)
|
)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def space_usable_for_service(service: Service) -> bool:
|
def service_snapshot_size(snapshot_id: str) -> int:
|
||||||
|
return Backups.provider().backuper.restored_size(
|
||||||
|
snapshot_id,
|
||||||
|
)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def space_usable_for_service(service: Service) -> int:
|
||||||
folders = service.get_folders()
|
folders = service.get_folders()
|
||||||
if folders == []:
|
if folders == []:
|
||||||
raise ValueError("unallocated service", service.get_id())
|
raise ValueError("unallocated service", service.get_id())
|
||||||
|
|
|
@ -26,14 +26,14 @@ class AbstractBackuper(ABC):
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def init(self, repo_name):
|
def init(self):
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def restore_from_backup(self, repo_name: str, snapshot_id: str, folders: List[str]):
|
def restore_from_backup(self, snapshot_id: str, folders: List[str]):
|
||||||
"""Restore a target folder using a snapshot"""
|
"""Restore a target folder using a snapshot"""
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def restored_size(self, repo_name, snapshot_id) -> float:
|
def restored_size(self, snapshot_id: str) -> int:
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
|
@ -18,12 +18,12 @@ class NoneBackupper(AbstractBackuper):
|
||||||
"""Get all snapshots from the repo"""
|
"""Get all snapshots from the repo"""
|
||||||
return []
|
return []
|
||||||
|
|
||||||
def init(self, repo_name):
|
def init(self):
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
def restore_from_backup(self, repo_name: str, snapshot_id: str, folders: List[str]):
|
def restore_from_backup(self, snapshot_id: str, folders: List[str]):
|
||||||
"""Restore a target folder using a snapshot"""
|
"""Restore a target folder using a snapshot"""
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
def restored_size(self, repo_name, snapshot_id) -> float:
|
def restored_size(self, snapshot_id: str) -> int:
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
|
@ -50,7 +50,7 @@ class ResticBackuper(AbstractBackuper):
|
||||||
def _password_command(self):
|
def _password_command(self):
|
||||||
return f"echo {LocalBackupSecret.get()}"
|
return f"echo {LocalBackupSecret.get()}"
|
||||||
|
|
||||||
def restic_command(self, *args, branch_name: str = ""):
|
def restic_command(self, *args, tag: str = ""):
|
||||||
command = [
|
command = [
|
||||||
"restic",
|
"restic",
|
||||||
"-o",
|
"-o",
|
||||||
|
@ -60,11 +60,11 @@ class ResticBackuper(AbstractBackuper):
|
||||||
"--password-command",
|
"--password-command",
|
||||||
self._password_command(),
|
self._password_command(),
|
||||||
]
|
]
|
||||||
if branch_name != "":
|
if tag != "":
|
||||||
command.extend(
|
command.extend(
|
||||||
[
|
[
|
||||||
"--tag",
|
"--tag",
|
||||||
branch_name,
|
tag,
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
if args != []:
|
if args != []:
|
||||||
|
@ -92,10 +92,10 @@ class ResticBackuper(AbstractBackuper):
|
||||||
universal_newlines=True,
|
universal_newlines=True,
|
||||||
) as handle:
|
) as handle:
|
||||||
for line in iter(handle.stdout.readline, ""):
|
for line in iter(handle.stdout.readline, ""):
|
||||||
if not "NOTICE:" in line:
|
if "NOTICE:" not in line:
|
||||||
yield line
|
yield line
|
||||||
|
|
||||||
def start_backup(self, folders: List[str], repo_name: str):
|
def start_backup(self, folders: List[str], tag: str):
|
||||||
"""
|
"""
|
||||||
Start backup with restic
|
Start backup with restic
|
||||||
"""
|
"""
|
||||||
|
@ -107,16 +107,16 @@ class ResticBackuper(AbstractBackuper):
|
||||||
"backup",
|
"backup",
|
||||||
"--json",
|
"--json",
|
||||||
folders,
|
folders,
|
||||||
branch_name=repo_name,
|
tag=tag,
|
||||||
)
|
)
|
||||||
|
|
||||||
messages = []
|
messages = []
|
||||||
job = get_backup_job(get_service_by_id(repo_name))
|
job = get_backup_job(get_service_by_id(tag))
|
||||||
try:
|
try:
|
||||||
for raw_message in ResticBackuper.output_yielder(backup_command):
|
for raw_message in ResticBackuper.output_yielder(backup_command):
|
||||||
message = self.parse_message(raw_message, job)
|
message = self.parse_message(raw_message, job)
|
||||||
messages.append(message)
|
messages.append(message)
|
||||||
return ResticBackuper._snapshot_from_backup_messages(messages, repo_name)
|
return ResticBackuper._snapshot_from_backup_messages(messages, tag)
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
raise ValueError("could not create a snapshot: ", messages) from e
|
raise ValueError("could not create a snapshot: ", messages) from e
|
||||||
|
|
||||||
|
@ -128,7 +128,7 @@ class ResticBackuper(AbstractBackuper):
|
||||||
raise ValueError("no summary message in restic json output")
|
raise ValueError("no summary message in restic json output")
|
||||||
|
|
||||||
def parse_message(self, raw_message, job=None) -> object:
|
def parse_message(self, raw_message, job=None) -> object:
|
||||||
message = self.parse_json_output(raw_message)
|
message = ResticBackuper.parse_json_output(raw_message)
|
||||||
if message["message_type"] == "status":
|
if message["message_type"] == "status":
|
||||||
if job is not None: # only update status if we run under some job
|
if job is not None: # only update status if we run under some job
|
||||||
Jobs.update(
|
Jobs.update(
|
||||||
|
@ -168,12 +168,12 @@ class ResticBackuper(AbstractBackuper):
|
||||||
|
|
||||||
with subprocess.Popen(command, stdout=subprocess.PIPE, shell=False) as handle:
|
with subprocess.Popen(command, stdout=subprocess.PIPE, shell=False) as handle:
|
||||||
output = handle.communicate()[0].decode("utf-8")
|
output = handle.communicate()[0].decode("utf-8")
|
||||||
if not self.has_json(output):
|
if not ResticBackuper.has_json(output):
|
||||||
return False
|
return False
|
||||||
# raise NotImplementedError("error(big): " + output)
|
# raise NotImplementedError("error(big): " + output)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def restored_size(self, repo_name, snapshot_id) -> float:
|
def restored_size(self, snapshot_id: str) -> int:
|
||||||
"""
|
"""
|
||||||
Size of a snapshot
|
Size of a snapshot
|
||||||
"""
|
"""
|
||||||
|
@ -183,15 +183,19 @@ class ResticBackuper(AbstractBackuper):
|
||||||
"--json",
|
"--json",
|
||||||
)
|
)
|
||||||
|
|
||||||
with subprocess.Popen(command, stdout=subprocess.PIPE, shell=False) as handle:
|
with subprocess.Popen(
|
||||||
|
command,
|
||||||
|
stdout=subprocess.PIPE,
|
||||||
|
shell=False,
|
||||||
|
) as handle:
|
||||||
output = handle.communicate()[0].decode("utf-8")
|
output = handle.communicate()[0].decode("utf-8")
|
||||||
try:
|
try:
|
||||||
parsed_output = self.parse_json_output(output)
|
parsed_output = ResticBackuper.parse_json_output(output)
|
||||||
return parsed_output["total_size"]
|
return parsed_output["total_size"]
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
raise ValueError("cannot restore a snapshot: " + output) from e
|
raise ValueError("cannot restore a snapshot: " + output) from e
|
||||||
|
|
||||||
def restore_from_backup(self, repo_name, snapshot_id, folders):
|
def restore_from_backup(self, snapshot_id, folders):
|
||||||
"""
|
"""
|
||||||
Restore from backup with restic
|
Restore from backup with restic
|
||||||
"""
|
"""
|
||||||
|
@ -235,7 +239,7 @@ class ResticBackuper(AbstractBackuper):
|
||||||
if "Is there a repository at the following location?" in output:
|
if "Is there a repository at the following location?" in output:
|
||||||
raise ValueError("No repository! : " + output)
|
raise ValueError("No repository! : " + output)
|
||||||
try:
|
try:
|
||||||
return self.parse_json_output(output)
|
return ResticBackuper.parse_json_output(output)
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
raise ValueError("Cannot load snapshots: ") from e
|
raise ValueError("Cannot load snapshots: ") from e
|
||||||
|
|
||||||
|
@ -252,8 +256,9 @@ class ResticBackuper(AbstractBackuper):
|
||||||
snapshots.append(snapshot)
|
snapshots.append(snapshot)
|
||||||
return snapshots
|
return snapshots
|
||||||
|
|
||||||
def parse_json_output(self, output: str) -> object:
|
@staticmethod
|
||||||
starting_index = self.json_start(output)
|
def parse_json_output(output: str) -> object:
|
||||||
|
starting_index = ResticBackuper.json_start(output)
|
||||||
|
|
||||||
if starting_index == -1:
|
if starting_index == -1:
|
||||||
raise ValueError("There is no json in the restic output : " + output)
|
raise ValueError("There is no json in the restic output : " + output)
|
||||||
|
@ -273,7 +278,8 @@ class ResticBackuper(AbstractBackuper):
|
||||||
result_array.append(json.loads(message))
|
result_array.append(json.loads(message))
|
||||||
return result_array
|
return result_array
|
||||||
|
|
||||||
def json_start(self, output: str) -> int:
|
@staticmethod
|
||||||
|
def json_start(output: str) -> int:
|
||||||
indices = [
|
indices = [
|
||||||
output.find("["),
|
output.find("["),
|
||||||
output.find("{"),
|
output.find("{"),
|
||||||
|
@ -284,7 +290,8 @@ class ResticBackuper(AbstractBackuper):
|
||||||
return -1
|
return -1
|
||||||
return min(indices)
|
return min(indices)
|
||||||
|
|
||||||
def has_json(self, output: str) -> bool:
|
@staticmethod
|
||||||
if self.json_start(output) == -1:
|
def has_json(output: str) -> bool:
|
||||||
|
if ResticBackuper.json_start(output) == -1:
|
||||||
return False
|
return False
|
||||||
return True
|
return True
|
||||||
|
|
|
@ -51,6 +51,8 @@ def add_backup_job(service: Service) -> Job:
|
||||||
|
|
||||||
def add_restore_job(snapshot: Snapshot) -> Job:
|
def add_restore_job(snapshot: Snapshot) -> Job:
|
||||||
service = get_service_by_id(snapshot.service_name)
|
service = get_service_by_id(snapshot.service_name)
|
||||||
|
if service is None:
|
||||||
|
raise ValueError(f"no such service: {snapshot.service_name}")
|
||||||
if is_something_queued_for(service):
|
if is_something_queued_for(service):
|
||||||
message = (
|
message = (
|
||||||
f"Cannot start a restore of {service.get_id()}, another operation is queued: "
|
f"Cannot start a restore of {service.get_id()}, another operation is queued: "
|
||||||
|
|
|
@ -1,23 +1,29 @@
|
||||||
from selfprivacy_api.graphql.queries.providers import BackupProvider
|
from typing import Type
|
||||||
|
|
||||||
|
from selfprivacy_api.graphql.queries.providers import (
|
||||||
|
BackupProvider as BackupProviderEnum,
|
||||||
|
)
|
||||||
from selfprivacy_api.backup.providers.provider import AbstractBackupProvider
|
from selfprivacy_api.backup.providers.provider import AbstractBackupProvider
|
||||||
|
|
||||||
from selfprivacy_api.backup.providers.backblaze import Backblaze
|
from selfprivacy_api.backup.providers.backblaze import Backblaze
|
||||||
from selfprivacy_api.backup.providers.memory import InMemoryBackup
|
from selfprivacy_api.backup.providers.memory import InMemoryBackup
|
||||||
from selfprivacy_api.backup.providers.local_file import LocalFileBackup
|
from selfprivacy_api.backup.providers.local_file import LocalFileBackup
|
||||||
|
from selfprivacy_api.backup.providers.none import NoBackups
|
||||||
|
|
||||||
PROVIDER_MAPPING = {
|
PROVIDER_MAPPING: dict[BackupProviderEnum, Type[AbstractBackupProvider]] = {
|
||||||
BackupProvider.BACKBLAZE: Backblaze,
|
BackupProviderEnum.BACKBLAZE: Backblaze,
|
||||||
BackupProvider.MEMORY: InMemoryBackup,
|
BackupProviderEnum.MEMORY: InMemoryBackup,
|
||||||
BackupProvider.FILE: LocalFileBackup,
|
BackupProviderEnum.FILE: LocalFileBackup,
|
||||||
BackupProvider.NONE: AbstractBackupProvider,
|
BackupProviderEnum.NONE: NoBackups,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
def get_provider(provider_type: BackupProvider) -> AbstractBackupProvider:
|
def get_provider(
|
||||||
|
provider_type: BackupProviderEnum,
|
||||||
|
) -> Type[AbstractBackupProvider]:
|
||||||
return PROVIDER_MAPPING[provider_type]
|
return PROVIDER_MAPPING[provider_type]
|
||||||
|
|
||||||
|
|
||||||
def get_kind(provider: AbstractBackupProvider) -> str:
|
def get_kind(provider: AbstractBackupProvider) -> str:
|
||||||
for key, value in PROVIDER_MAPPING.items():
|
"""Get the kind of the provider in the form of a string"""
|
||||||
if isinstance(provider, value):
|
return provider.name.value
|
||||||
return key.value
|
|
||||||
|
|
|
@ -1,8 +1,13 @@
|
||||||
from .provider import AbstractBackupProvider
|
from .provider import AbstractBackupProvider
|
||||||
from selfprivacy_api.backup.backuppers.restic_backupper import ResticBackuper
|
from selfprivacy_api.backup.backuppers.restic_backupper import ResticBackuper
|
||||||
|
from selfprivacy_api.graphql.queries.providers import (
|
||||||
|
BackupProvider as BackupProviderEnum,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class Backblaze(AbstractBackupProvider):
|
class Backblaze(AbstractBackupProvider):
|
||||||
backuper = ResticBackuper("--b2-account", "--b2-key", ":b2:")
|
@property
|
||||||
|
def backuper(self):
|
||||||
|
return ResticBackuper("--b2-account", "--b2-key", ":b2:")
|
||||||
|
|
||||||
name = "BACKBLAZE"
|
name = BackupProviderEnum.BACKBLAZE
|
||||||
|
|
|
@ -1,7 +1,13 @@
|
||||||
from .provider import AbstractBackupProvider
|
from .provider import AbstractBackupProvider
|
||||||
from selfprivacy_api.backup.backuppers.restic_backupper import ResticBackuper
|
from selfprivacy_api.backup.backuppers.restic_backupper import ResticBackuper
|
||||||
|
from selfprivacy_api.graphql.queries.providers import (
|
||||||
|
BackupProvider as BackupProviderEnum,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class LocalFileBackup(AbstractBackupProvider):
|
class LocalFileBackup(AbstractBackupProvider):
|
||||||
backuper = ResticBackuper("", "", ":local:")
|
@property
|
||||||
name = "FILE"
|
def backuper(self):
|
||||||
|
return ResticBackuper("", "", ":local:")
|
||||||
|
|
||||||
|
name = BackupProviderEnum.FILE
|
||||||
|
|
|
@ -1,8 +1,13 @@
|
||||||
from .provider import AbstractBackupProvider
|
from .provider import AbstractBackupProvider
|
||||||
from selfprivacy_api.backup.backuppers.restic_backupper import ResticBackuper
|
from selfprivacy_api.backup.backuppers.restic_backupper import ResticBackuper
|
||||||
|
from selfprivacy_api.graphql.queries.providers import (
|
||||||
|
BackupProvider as BackupProviderEnum,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class InMemoryBackup(AbstractBackupProvider):
|
class InMemoryBackup(AbstractBackupProvider):
|
||||||
backuper = ResticBackuper("", "", ":memory:")
|
@property
|
||||||
|
def backuper(self):
|
||||||
|
return ResticBackuper("", "", ":memory:")
|
||||||
|
|
||||||
name = "MEMORY"
|
name = BackupProviderEnum.MEMORY
|
||||||
|
|
13
selfprivacy_api/backup/providers/none.py
Normal file
13
selfprivacy_api/backup/providers/none.py
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
from .provider import AbstractBackupProvider
|
||||||
|
from selfprivacy_api.backup.backuppers.none_backupper import NoneBackupper
|
||||||
|
from selfprivacy_api.graphql.queries.providers import (
|
||||||
|
BackupProvider as BackupProviderEnum,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class NoBackups(AbstractBackupProvider):
|
||||||
|
@property
|
||||||
|
def backuper(self):
|
||||||
|
return NoneBackupper()
|
||||||
|
|
||||||
|
name = BackupProviderEnum.NONE
|
|
@ -1,19 +1,22 @@
|
||||||
"""
|
"""
|
||||||
An abstract class for BackBlaze, S3 etc.
|
An abstract class for BackBlaze, S3 etc.
|
||||||
It assumes that while some providers are supported via restic/rclone, others may
|
It assumes that while some providers are supported via restic/rclone, others
|
||||||
require different backends
|
may require different backends
|
||||||
"""
|
"""
|
||||||
from abc import ABC
|
from abc import ABC, abstractmethod
|
||||||
from selfprivacy_api.backup.backuppers import AbstractBackuper
|
from selfprivacy_api.backup.backuppers import AbstractBackuper
|
||||||
from selfprivacy_api.backup.backuppers.none_backupper import NoneBackupper
|
from selfprivacy_api.graphql.queries.providers import (
|
||||||
|
BackupProvider as BackupProviderEnum,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class AbstractBackupProvider(ABC):
|
class AbstractBackupProvider(ABC):
|
||||||
@property
|
@property
|
||||||
|
@abstractmethod
|
||||||
def backuper(self) -> AbstractBackuper:
|
def backuper(self) -> AbstractBackuper:
|
||||||
return NoneBackupper()
|
raise NotImplementedError
|
||||||
|
|
||||||
name = "NONE"
|
name: BackupProviderEnum
|
||||||
|
|
||||||
def __init__(self, login="", key="", location="", repo_id=""):
|
def __init__(self, login="", key="", location="", repo_id=""):
|
||||||
self.backuper.set_creds(login, key, location)
|
self.backuper.set_creds(login, key, location)
|
||||||
|
|
|
@ -5,7 +5,10 @@ from selfprivacy_api.models.backup.snapshot import Snapshot
|
||||||
from selfprivacy_api.models.backup.provider import BackupProviderModel
|
from selfprivacy_api.models.backup.provider import BackupProviderModel
|
||||||
|
|
||||||
from selfprivacy_api.utils.redis_pool import RedisPool
|
from selfprivacy_api.utils.redis_pool import RedisPool
|
||||||
from selfprivacy_api.utils.redis_model_storage import store_model_as_hash, hash_as_model
|
from selfprivacy_api.utils.redis_model_storage import (
|
||||||
|
store_model_as_hash,
|
||||||
|
hash_as_model,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
from selfprivacy_api.services.service import Service
|
from selfprivacy_api.services.service import Service
|
||||||
|
@ -153,8 +156,12 @@ class Storage:
|
||||||
)
|
)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def load_provider() -> BackupProviderModel:
|
def load_provider() -> Optional[BackupProviderModel]:
|
||||||
provider_model = hash_as_model(redis, REDIS_PROVIDER_KEY, BackupProviderModel)
|
provider_model = hash_as_model(
|
||||||
|
redis,
|
||||||
|
REDIS_PROVIDER_KEY,
|
||||||
|
BackupProviderModel,
|
||||||
|
)
|
||||||
return provider_model
|
return provider_model
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
|
|
|
@ -49,7 +49,7 @@ class BackupMutations:
|
||||||
) -> GenericBackupConfigReturn:
|
) -> GenericBackupConfigReturn:
|
||||||
"""Initialize a new repository"""
|
"""Initialize a new repository"""
|
||||||
Backups.set_provider(
|
Backups.set_provider(
|
||||||
kind=repository.provider.value,
|
kind=repository.provider,
|
||||||
login=repository.login,
|
login=repository.login,
|
||||||
key=repository.password,
|
key=repository.password,
|
||||||
location=repository.location_name,
|
location=repository.location_name,
|
||||||
|
@ -57,7 +57,10 @@ class BackupMutations:
|
||||||
)
|
)
|
||||||
Backups.init_repo()
|
Backups.init_repo()
|
||||||
return GenericBackupConfigReturn(
|
return GenericBackupConfigReturn(
|
||||||
success=True, message="", code="200", configuration=Backup().configuration()
|
success=True,
|
||||||
|
message="",
|
||||||
|
code="200",
|
||||||
|
configuration=Backup().configuration(),
|
||||||
)
|
)
|
||||||
|
|
||||||
@strawberry.mutation(permission_classes=[IsAuthenticated])
|
@strawberry.mutation(permission_classes=[IsAuthenticated])
|
||||||
|
@ -65,7 +68,10 @@ class BackupMutations:
|
||||||
"""Remove repository"""
|
"""Remove repository"""
|
||||||
Backups.reset()
|
Backups.reset()
|
||||||
return GenericBackupConfigReturn(
|
return GenericBackupConfigReturn(
|
||||||
success=True, message="", code="200", configuration=Backup().configuration()
|
success=True,
|
||||||
|
message="",
|
||||||
|
code="200",
|
||||||
|
configuration=Backup().configuration(),
|
||||||
)
|
)
|
||||||
|
|
||||||
@strawberry.mutation(permission_classes=[IsAuthenticated])
|
@strawberry.mutation(permission_classes=[IsAuthenticated])
|
||||||
|
@ -79,7 +85,10 @@ class BackupMutations:
|
||||||
Backups.set_autobackup_period_minutes(0)
|
Backups.set_autobackup_period_minutes(0)
|
||||||
|
|
||||||
return GenericBackupConfigReturn(
|
return GenericBackupConfigReturn(
|
||||||
success=True, message="", code="200", configuration=Backup().configuration()
|
success=True,
|
||||||
|
message="",
|
||||||
|
code="200",
|
||||||
|
configuration=Backup().configuration(),
|
||||||
)
|
)
|
||||||
|
|
||||||
@strawberry.mutation(permission_classes=[IsAuthenticated])
|
@strawberry.mutation(permission_classes=[IsAuthenticated])
|
||||||
|
@ -97,36 +106,52 @@ class BackupMutations:
|
||||||
|
|
||||||
job = add_backup_job(service)
|
job = add_backup_job(service)
|
||||||
start_backup(service)
|
start_backup(service)
|
||||||
job = job_to_api_job(job)
|
|
||||||
|
|
||||||
return GenericJobMutationReturn(
|
return GenericJobMutationReturn(
|
||||||
success=True,
|
success=True,
|
||||||
code=200,
|
code=200,
|
||||||
message="Backup job queued",
|
message="Backup job queued",
|
||||||
job=job,
|
job=job_to_api_job(job),
|
||||||
)
|
)
|
||||||
|
|
||||||
@strawberry.mutation(permission_classes=[IsAuthenticated])
|
@strawberry.mutation(permission_classes=[IsAuthenticated])
|
||||||
def restore_backup(self, snapshot_id: str) -> GenericJobMutationReturn:
|
def restore_backup(self, snapshot_id: str) -> GenericJobMutationReturn:
|
||||||
"""Restore backup"""
|
"""Restore backup"""
|
||||||
snap = Backups.get_snapshot_by_id(snapshot_id)
|
snap = Backups.get_snapshot_by_id(snapshot_id)
|
||||||
service = get_service_by_id(snap.service_name)
|
|
||||||
if snap is None:
|
if snap is None:
|
||||||
return GenericJobMutationReturn(
|
return GenericJobMutationReturn(
|
||||||
success=False,
|
success=False,
|
||||||
code=400,
|
code=404,
|
||||||
message=f"No such snapshot: {snapshot_id}",
|
message=f"No such snapshot: {snapshot_id}",
|
||||||
job=None,
|
job=None,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
service = get_service_by_id(snap.service_name)
|
||||||
|
if service is None:
|
||||||
|
return GenericJobMutationReturn(
|
||||||
|
success=False,
|
||||||
|
code=404,
|
||||||
|
message=f"nonexistent service: {snap.service_name}",
|
||||||
|
job=None,
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
job = add_restore_job(snap)
|
job = add_restore_job(snap)
|
||||||
|
except ValueError as e:
|
||||||
|
return GenericJobMutationReturn(
|
||||||
|
success=False,
|
||||||
|
code=400,
|
||||||
|
message=str(e),
|
||||||
|
job=None,
|
||||||
|
)
|
||||||
|
|
||||||
restore_snapshot(snap)
|
restore_snapshot(snap)
|
||||||
|
|
||||||
return GenericJobMutationReturn(
|
return GenericJobMutationReturn(
|
||||||
success=True,
|
success=True,
|
||||||
code=200,
|
code=200,
|
||||||
message="restore job created",
|
message="restore job created",
|
||||||
job=job,
|
job=job_to_api_job(job),
|
||||||
)
|
)
|
||||||
|
|
||||||
@strawberry.mutation(permission_classes=[IsAuthenticated])
|
@strawberry.mutation(permission_classes=[IsAuthenticated])
|
||||||
|
|
|
@ -73,7 +73,7 @@ def dummy_service(tmpdir, backups, raw_dummy_service):
|
||||||
assert not path.exists(repo_path)
|
assert not path.exists(repo_path)
|
||||||
# assert not repo_path
|
# assert not repo_path
|
||||||
|
|
||||||
Backups.init_repo(service)
|
Backups.init_repo()
|
||||||
|
|
||||||
# register our service
|
# register our service
|
||||||
services.services.append(service)
|
services.services.append(service)
|
||||||
|
@ -232,7 +232,7 @@ def test_restore(backups, dummy_service):
|
||||||
def test_sizing(backups, dummy_service):
|
def test_sizing(backups, dummy_service):
|
||||||
Backups.back_up(dummy_service)
|
Backups.back_up(dummy_service)
|
||||||
snap = Backups.get_snapshots(dummy_service)[0]
|
snap = Backups.get_snapshots(dummy_service)[0]
|
||||||
size = Backups.service_snapshot_size(dummy_service, snap.id)
|
size = Backups.service_snapshot_size(snap.id)
|
||||||
assert size is not None
|
assert size is not None
|
||||||
assert size > 0
|
assert size > 0
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue