2022-02-20 06:04:12 +00:00
|
|
|
from datetime import datetime as dt
|
|
|
|
from datetime import timedelta as td
|
|
|
|
from functools import wraps
|
|
|
|
from unittest.mock import MagicMock
|
2023-05-01 08:28:11 +00:00
|
|
|
|
2021-02-17 02:14:58 +00:00
|
|
|
import pytest
|
2024-10-18 11:08:59 +00:00
|
|
|
from pytest import mark
|
|
|
|
|
2018-11-30 00:40:18 +00:00
|
|
|
import vorta.borg
|
2022-02-20 06:04:12 +00:00
|
|
|
import vorta.scheduler
|
2022-05-21 11:06:10 +00:00
|
|
|
from vorta.scheduler import ScheduleStatus, ScheduleStatusType, VortaScheduler
|
2022-02-20 06:04:12 +00:00
|
|
|
from vorta.store.models import BackupProfileModel, EventLogModel
|
|
|
|
|
|
|
|
PROFILE_NAME = 'Default'
|
|
|
|
FIXED_SCHEDULE = 'fixed'
|
|
|
|
INTERVAL_SCHEDULE = 'interval'
|
|
|
|
MANUAL_SCHEDULE = 'off'
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def clockmock(monkeypatch):
|
|
|
|
datetime_mock = MagicMock(wraps=dt)
|
|
|
|
monkeypatch.setattr(vorta.scheduler, "dt", datetime_mock)
|
|
|
|
|
|
|
|
return datetime_mock
|
|
|
|
|
2018-11-30 00:40:18 +00:00
|
|
|
|
2022-02-20 06:04:12 +00:00
|
|
|
def prepare(func):
|
|
|
|
"""Decorator adding common preparation steps."""
|
2022-08-15 17:02:40 +00:00
|
|
|
|
2022-02-20 06:04:12 +00:00
|
|
|
@wraps(func)
|
|
|
|
def do(qapp, qtbot, mocker, borg_json_output):
|
|
|
|
stdout, stderr = borg_json_output('create')
|
2022-08-15 17:02:40 +00:00
|
|
|
popen_result = mocker.MagicMock(stdout=stdout, stderr=stderr, returncode=0)
|
|
|
|
mocker.patch.object(vorta.borg.borg_job, 'Popen', return_value=popen_result)
|
2018-11-30 00:40:18 +00:00
|
|
|
|
2022-02-20 06:04:12 +00:00
|
|
|
return func(qapp, qtbot, mocker, borg_json_output)
|
|
|
|
|
|
|
|
return do
|
|
|
|
|
|
|
|
|
|
|
|
@prepare
|
2020-03-03 05:19:36 +00:00
|
|
|
def test_scheduler_create_backup(qapp, qtbot, mocker, borg_json_output):
|
2022-02-20 06:04:12 +00:00
|
|
|
"""Test running a backup with `create_backup`."""
|
2021-11-12 07:05:31 +00:00
|
|
|
events_before = EventLogModel.select().count()
|
2018-11-30 00:40:18 +00:00
|
|
|
|
2021-11-12 07:05:31 +00:00
|
|
|
with qtbot.waitSignal(qapp.backup_finished_event, **pytest._wait_defaults):
|
|
|
|
qapp.scheduler.create_backup(1)
|
|
|
|
|
2022-05-21 11:06:10 +00:00
|
|
|
assert EventLogModel.select().count() == events_before + 1
|
2022-02-20 06:04:12 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_manual_mode():
|
|
|
|
"""Test scheduling in manual mode."""
|
|
|
|
scheduler = VortaScheduler()
|
|
|
|
|
|
|
|
# setup model
|
|
|
|
profile = BackupProfileModel.get(name=PROFILE_NAME)
|
|
|
|
profile.schedule_make_up_missed = False
|
|
|
|
profile.schedule_mode = MANUAL_SCHEDULE
|
|
|
|
profile.save()
|
|
|
|
|
|
|
|
# test
|
|
|
|
scheduler.set_timer_for_profile(profile.id)
|
|
|
|
assert len(scheduler.timers) == 0
|
|
|
|
|
|
|
|
|
|
|
|
def test_simple_schedule(clockmock):
|
|
|
|
"""Test a simple scheduling including `next_job` and `remove_job`."""
|
|
|
|
scheduler = VortaScheduler()
|
|
|
|
|
|
|
|
# setup
|
2022-05-21 11:06:10 +00:00
|
|
|
time = dt(2020, 5, 6, 4, 30)
|
|
|
|
clockmock.now.return_value = time
|
2022-02-20 06:04:12 +00:00
|
|
|
|
|
|
|
profile = BackupProfileModel.get(name=PROFILE_NAME)
|
|
|
|
profile.schedule_make_up_missed = False
|
|
|
|
profile.schedule_mode = INTERVAL_SCHEDULE
|
|
|
|
profile.schedule_interval_unit = 'hours'
|
|
|
|
profile.schedule_interval_count = 3
|
|
|
|
profile.save()
|
|
|
|
|
2022-08-15 17:02:40 +00:00
|
|
|
event = EventLogModel(
|
|
|
|
subcommand='create', profile=profile.id, returncode=0, category='scheduled', start_time=time, end_time=time
|
|
|
|
)
|
2022-05-21 11:06:10 +00:00
|
|
|
event.save()
|
|
|
|
|
2022-02-20 06:04:12 +00:00
|
|
|
# test set timer and next_job
|
|
|
|
scheduler.set_timer_for_profile(profile.id)
|
|
|
|
assert len(scheduler.timers) == 1
|
|
|
|
assert scheduler.next_job() == '07:30 ({})'.format(PROFILE_NAME)
|
2022-05-21 11:06:10 +00:00
|
|
|
assert scheduler.next_job_for_profile(profile.id) == ScheduleStatus(
|
2022-08-15 17:02:40 +00:00
|
|
|
ScheduleStatusType.SCHEDULED, dt(2020, 5, 6, 7, 30)
|
|
|
|
)
|
2022-02-20 06:04:12 +00:00
|
|
|
|
|
|
|
# test remove_job and next_job
|
|
|
|
scheduler.remove_job(profile.id)
|
|
|
|
assert len(scheduler.timers) == 0
|
|
|
|
assert scheduler.next_job() == 'None scheduled'
|
2022-05-21 11:06:10 +00:00
|
|
|
assert scheduler.next_job_for_profile(profile.id) == ScheduleStatus(ScheduleStatusType.UNSCHEDULED)
|
2022-02-20 06:04:12 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mark.parametrize("scheduled", [True, False])
|
|
|
|
@mark.parametrize(
|
|
|
|
"passed_time, now, unit, count, added_time",
|
|
|
|
[
|
|
|
|
# simple
|
|
|
|
(td(), td(hours=4, minutes=30), 'hours', 3, td(hours=3)),
|
|
|
|
# next day
|
|
|
|
(td(), td(hours=4, minutes=30), 'hours', 20, td(hours=20)),
|
|
|
|
# passed by less than interval
|
|
|
|
(td(hours=2), td(hours=4, minutes=30), 'hours', 3, td(hours=1)),
|
|
|
|
# passed by exactly interval
|
|
|
|
(td(hours=3), td(hours=4, minutes=30), 'hours', 3, td(hours=3)),
|
|
|
|
# passed by multiple times the interval
|
2022-08-15 17:02:40 +00:00
|
|
|
(td(hours=7), td(hours=4, minutes=30), 'hours', 3, td(hours=2)),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
def test_interval(clockmock, passed_time, scheduled, now, unit, count, added_time):
|
2022-02-20 06:04:12 +00:00
|
|
|
"""Test scheduling in interval mode."""
|
|
|
|
# setup
|
|
|
|
scheduler = VortaScheduler()
|
|
|
|
|
|
|
|
time = dt(2020, 5, 4, 0, 0) + now
|
|
|
|
clockmock.now.return_value = time
|
|
|
|
|
|
|
|
profile = BackupProfileModel.get(name=PROFILE_NAME)
|
|
|
|
profile.schedule_make_up_missed = False
|
|
|
|
profile.schedule_mode = INTERVAL_SCHEDULE
|
|
|
|
profile.schedule_interval_unit = unit
|
|
|
|
profile.schedule_interval_count = count
|
|
|
|
profile.save()
|
|
|
|
|
2022-08-15 17:02:40 +00:00
|
|
|
event = EventLogModel(
|
|
|
|
subcommand='create',
|
|
|
|
profile=profile.id,
|
|
|
|
returncode=0,
|
|
|
|
category='scheduled' if scheduled else '',
|
|
|
|
start_time=time - passed_time,
|
|
|
|
end_time=time - passed_time,
|
|
|
|
)
|
2022-02-20 06:04:12 +00:00
|
|
|
event.save()
|
|
|
|
|
|
|
|
# run test
|
|
|
|
scheduler.set_timer_for_profile(profile.id)
|
|
|
|
assert scheduler.timers[profile.id]['dt'] == time + added_time
|
|
|
|
|
|
|
|
|
|
|
|
@mark.parametrize("scheduled", [True, False])
|
2022-08-15 17:02:40 +00:00
|
|
|
@mark.parametrize("passed_time", [td(hours=0), td(hours=5), td(hours=14), td(hours=27)])
|
2022-02-20 06:04:12 +00:00
|
|
|
@mark.parametrize(
|
|
|
|
"now, hour, minute",
|
|
|
|
[
|
|
|
|
# same day
|
|
|
|
(td(hours=4, minutes=30), 15, 00),
|
|
|
|
# next day
|
|
|
|
(td(hours=4, minutes=30), 3, 30),
|
2022-08-15 17:02:40 +00:00
|
|
|
],
|
|
|
|
)
|
2022-02-20 06:04:12 +00:00
|
|
|
def test_fixed(clockmock, passed_time, scheduled, now, hour, minute):
|
|
|
|
"""Test scheduling in fixed mode."""
|
|
|
|
# setup
|
|
|
|
scheduler = VortaScheduler()
|
|
|
|
|
|
|
|
time = dt(2020, 5, 4, 0, 0) + now
|
|
|
|
clockmock.now.return_value = time
|
|
|
|
|
|
|
|
profile = BackupProfileModel.get(name=PROFILE_NAME)
|
|
|
|
profile.schedule_make_up_missed = False
|
|
|
|
profile.schedule_mode = FIXED_SCHEDULE
|
|
|
|
profile.schedule_fixed_hour = hour
|
|
|
|
profile.schedule_fixed_minute = minute
|
|
|
|
profile.save()
|
|
|
|
|
|
|
|
last_time = time - passed_time
|
2022-08-15 17:02:40 +00:00
|
|
|
event = EventLogModel(
|
|
|
|
subcommand='create',
|
|
|
|
profile=profile.id,
|
|
|
|
returncode=0,
|
|
|
|
category='scheduled' if scheduled else '',
|
|
|
|
start_time=last_time,
|
|
|
|
end_time=last_time,
|
|
|
|
)
|
2022-02-20 06:04:12 +00:00
|
|
|
event.save()
|
|
|
|
|
|
|
|
# run test
|
|
|
|
expected = time.replace(hour=hour, minute=minute)
|
|
|
|
|
|
|
|
if time >= expected or last_time.date() == expected.date():
|
|
|
|
expected += td(days=1)
|
|
|
|
|
|
|
|
scheduler.set_timer_for_profile(profile.id)
|
|
|
|
assert scheduler.timers[profile.id]['dt'] == expected
|