From 449b95e1b6707ea9b7359b2ee9023f22476a5189 Mon Sep 17 00:00:00 2001
From: Shivansh Singh <89853707+shivansh02@users.noreply.github.com>
Date: Sat, 29 Jun 2024 15:08:02 +0530
Subject: [PATCH] Add compaction scheduling. By @shivansh02 (#1981)
---
src/vorta/assets/UI/scheduletab.ui | 110 +++++++++++++++++++++++++++++
src/vorta/scheduler.py | 24 +++++++
src/vorta/store/connection.py | 2 +-
src/vorta/store/migrations.py | 16 +++++
src/vorta/store/models.py | 2 +
src/vorta/views/schedule_tab.py | 14 ++++
6 files changed, 167 insertions(+), 1 deletion(-)
diff --git a/src/vorta/assets/UI/scheduletab.ui b/src/vorta/assets/UI/scheduletab.ui
index f7cb7b34e..062c07717 100644
--- a/src/vorta/assets/UI/scheduletab.ui
+++ b/src/vorta/assets/UI/scheduletab.ui
@@ -443,6 +443,116 @@
+ -
+
+
+ Compaction:
+
+
+
+ -
+
+
+ 4
+
+
+ 4
+
+
-
+
+
+ Compact repository
+
+
+
+ -
+
+
+
+ 0
+ 0
+
+
+
+
+ 4
+
+
+ 0
+
+
+ 0
+
+
+ 0
+
+
+ 0
+
+
-
+
+
+ Qt::Horizontal
+
+
+ QSizePolicy::Fixed
+
+
+
+ 20
+ 20
+
+
+
+
+ -
+
+
+ Interval:
+
+
+
+ -
+
+
+ Qt::Horizontal
+
+
+ QSizePolicy::Fixed
+
+
+
+ 4
+ 1
+
+
+
+
+ -
+
+
+ 1
+
+
+ 52
+
+
+ 3
+
+
+
+ -
+
+
+ weeks
+
+
+
+
+
+
+
+
-
diff --git a/src/vorta/scheduler.py b/src/vorta/scheduler.py
index 82f4f0a9e..9ff19681c 100644
--- a/src/vorta/scheduler.py
+++ b/src/vorta/scheduler.py
@@ -5,18 +5,21 @@
from datetime import timedelta
from typing import Dict, NamedTuple, Optional, Tuple, Union
+from packaging import version
from PyQt6 import QtCore, QtDBus
from PyQt6.QtCore import QTimer
from PyQt6.QtWidgets import QApplication
from vorta import application
from vorta.borg.check import BorgCheckJob
+from vorta.borg.compact import BorgCompactJob
from vorta.borg.create import BorgCreateJob
from vorta.borg.list_repo import BorgListRepoJob
from vorta.borg.prune import BorgPruneJob
from vorta.i18n import translate
from vorta.notifications import VortaNotifications
from vorta.store.models import BackupProfileModel, EventLogModel
+from vorta.utils import borg_compat
logger = logging.getLogger(__name__)
@@ -489,6 +492,27 @@ def post_backup_tasks(self, profile_id):
job = BorgCheckJob(msg['cmd'], msg, profile.repo.id)
self.app.jobs_manager.add_job(job)
+ compaction_cutoff = dt.now() - timedelta(days=7 * profile.compaction_weeks)
+ recent_compactions = (
+ EventLogModel.select()
+ .where(
+ (EventLogModel.subcommand == '--info')
+ & (EventLogModel.start_time > compaction_cutoff)
+ & (EventLogModel.repo_url == profile.repo.url)
+ )
+ .count()
+ )
+
+ if (
+ profile.compaction_on
+ and recent_compactions == 0
+ and version.parse(borg_compat.version) >= version.parse("1.2")
+ ):
+ msg = BorgCompactJob.prepare(profile)
+ if msg['ok']:
+ job = BorgCompactJob(msg['cmd'], msg, profile.repo.id)
+ self.app.jobs_manager.add_job(job)
+
logger.info('Finished background task for profile %s', profile.name)
notifier.deliver(
self.tr('Vorta Backup'),
diff --git a/src/vorta/store/connection.py b/src/vorta/store/connection.py
index e02efe96b..a4401c98e 100644
--- a/src/vorta/store/connection.py
+++ b/src/vorta/store/connection.py
@@ -24,7 +24,7 @@
)
from .settings import get_misc_settings
-SCHEMA_VERSION = 22
+SCHEMA_VERSION = 23
@signals.post_save(sender=SettingsModel)
diff --git a/src/vorta/store/migrations.py b/src/vorta/store/migrations.py
index a26892597..76ebb6b2b 100644
--- a/src/vorta/store/migrations.py
+++ b/src/vorta/store/migrations.py
@@ -250,6 +250,22 @@ def run_migrations(current_schema, db_connection):
),
)
+ if current_schema.version < 23:
+ _apply_schema_update(
+ current_schema,
+ 23,
+ migrator.add_column(
+ BackupProfileModel._meta.table_name,
+ 'compaction_on',
+ pw.BooleanField(default=True),
+ ),
+ migrator.add_column(
+ BackupProfileModel._meta.table_name,
+ 'compaction_weeks',
+ pw.IntegerField(default=3),
+ ),
+ )
+
def _apply_schema_update(current_schema, version_after, *operations):
with DB.atomic():
diff --git a/src/vorta/store/models.py b/src/vorta/store/models.py
index d2117a10f..e7d08f305 100644
--- a/src/vorta/store/models.py
+++ b/src/vorta/store/models.py
@@ -90,6 +90,8 @@ class BackupProfileModel(BaseModel):
schedule_make_up_missed = pw.BooleanField(default=True)
validation_on = pw.BooleanField(default=True)
validation_weeks = pw.IntegerField(default=3)
+ compaction_on = pw.BooleanField(default=False)
+ compaction_weeks = pw.IntegerField(default=3)
prune_on = pw.BooleanField(default=False)
prune_hour = pw.IntegerField(default=2)
prune_day = pw.IntegerField(default=7)
diff --git a/src/vorta/views/schedule_tab.py b/src/vorta/views/schedule_tab.py
index dc97cfa47..2eba60fc6 100644
--- a/src/vorta/views/schedule_tab.py
+++ b/src/vorta/views/schedule_tab.py
@@ -63,10 +63,12 @@ def __init__(self, parent=None):
self.framePeriodic.setEnabled(False)
self.frameDaily.setEnabled(False)
self.frameValidation.setEnabled(False)
+ self.frameCompaction.setEnabled(False)
self.scheduleIntervalRadio.toggled.connect(self.framePeriodic.setEnabled)
self.scheduleFixedRadio.toggled.connect(self.frameDaily.setEnabled)
self.validationCheckBox.toggled.connect(self.frameValidation.setEnabled)
+ self.compactionCheckBox.toggled.connect(self.frameCompaction.setEnabled)
# POPULATE with data
self.populate_from_profile()
@@ -105,6 +107,12 @@ def __init__(self, parent=None):
self.validationWeeksCount.valueChanged.connect(
lambda new_val, attr='validation_weeks': self.save_profile_attr(attr, new_val)
)
+ self.compactionCheckBox.stateChanged.connect(
+ lambda new_val, attr='compaction_on': self.save_profile_attr(attr, new_val)
+ )
+ self.compactionWeeksCount.valueChanged.connect(
+ lambda new_val, attr='compaction_weeks': self.save_profile_attr(attr, new_val)
+ )
# Connect to schedule update
self.app.scheduler.schedule_changed.connect(lambda pid: self.draw_next_scheduled_backup())
@@ -154,6 +162,12 @@ def populate_from_profile(self):
)
self.validationWeeksCount.setValue(profile.validation_weeks)
+ # set borg compact options
+ self.compactionCheckBox.setCheckState(
+ QtCore.Qt.CheckState.Checked if profile.compaction_on else QtCore.Qt.CheckState.Unchecked
+ )
+ self.compactionWeeksCount.setValue(profile.compaction_weeks)
+
# Other checkbox options
self.pruneCheckBox.setCheckState(
QtCore.Qt.CheckState.Checked if profile.prune_on else QtCore.Qt.CheckState.Unchecked