2024-06-07 14:08:05 -04:00
|
|
|
import os
|
2024-09-16 14:54:43 -04:00
|
|
|
import json
|
2024-03-08 22:06:33 -05:00
|
|
|
import datetime
|
|
|
|
import uuid
|
|
|
|
import peewee
|
|
|
|
import logging
|
|
|
|
|
2024-05-26 13:45:13 -04:00
|
|
|
|
|
|
|
from app.classes.shared.helpers import Helpers
|
2024-03-08 22:06:33 -05:00
|
|
|
from app.classes.shared.console import Console
|
2024-06-14 14:17:00 -05:00
|
|
|
from app.classes.shared.migration import Migrator
|
2024-06-07 14:08:05 -04:00
|
|
|
from app.classes.shared.file_helpers import FileHelpers
|
2024-03-08 22:06:33 -05:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2024-08-13 00:12:22 -04:00
|
|
|
def is_valid_entry(entry, all_servers):
|
2024-07-29 18:04:01 -04:00
|
|
|
try:
|
2024-08-13 00:12:22 -04:00
|
|
|
return str(entry.server_id) in all_servers
|
2024-07-29 18:04:01 -04:00
|
|
|
except (TypeError, peewee.DoesNotExist):
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2024-03-08 22:06:33 -05:00
|
|
|
def migrate(migrator: Migrator, database, **kwargs):
|
|
|
|
"""
|
|
|
|
Write your migrations here.
|
|
|
|
"""
|
2024-09-16 17:37:04 -04:00
|
|
|
backup_migration_status = True
|
|
|
|
schedule_migration_status = True
|
2024-03-08 22:06:33 -05:00
|
|
|
db = database
|
2024-05-26 13:45:13 -04:00
|
|
|
Console.info("Starting Backups migrations")
|
|
|
|
Console.info(
|
|
|
|
"Migrations: Adding columns [backup_id, "
|
|
|
|
"backup_name, backup_location, enabled, default, action_id, backup_status]"
|
|
|
|
)
|
2024-05-28 19:42:02 -04:00
|
|
|
migrator.add_columns(
|
|
|
|
"backups",
|
2024-05-28 19:48:11 -04:00
|
|
|
backup_id=peewee.CharField(default=Helpers.create_uuid),
|
2024-05-28 19:42:02 -04:00
|
|
|
)
|
2024-03-08 22:06:33 -05:00
|
|
|
migrator.add_columns("backups", backup_name=peewee.CharField(default="Default"))
|
|
|
|
migrator.add_columns("backups", backup_location=peewee.CharField(default=""))
|
2024-05-25 13:51:40 -04:00
|
|
|
migrator.add_columns("backups", enabled=peewee.BooleanField(default=True))
|
2024-05-26 13:45:13 -04:00
|
|
|
migrator.add_columns("backups", default=peewee.BooleanField(default=False))
|
|
|
|
migrator.add_columns(
|
|
|
|
"backups",
|
|
|
|
status=peewee.CharField(default='{"status": "Standby", "message": ""}'),
|
|
|
|
)
|
2024-05-25 13:51:40 -04:00
|
|
|
migrator.add_columns(
|
|
|
|
"schedules", action_id=peewee.CharField(null=True, default=None)
|
|
|
|
)
|
2024-03-08 22:06:33 -05:00
|
|
|
|
|
|
|
class Servers(peewee.Model):
|
|
|
|
server_id = peewee.CharField(primary_key=True, default=str(uuid.uuid4()))
|
|
|
|
created = peewee.DateTimeField(default=datetime.datetime.now)
|
|
|
|
server_name = peewee.CharField(default="Server", index=True)
|
|
|
|
path = peewee.CharField(default="")
|
|
|
|
backup_path = peewee.CharField(default="")
|
|
|
|
executable = peewee.CharField(default="")
|
|
|
|
log_path = peewee.CharField(default="")
|
|
|
|
execution_command = peewee.CharField(default="")
|
|
|
|
auto_start = peewee.BooleanField(default=0)
|
|
|
|
auto_start_delay = peewee.IntegerField(default=10)
|
|
|
|
crash_detection = peewee.BooleanField(default=0)
|
|
|
|
stop_command = peewee.CharField(default="stop")
|
|
|
|
executable_update_url = peewee.CharField(default="")
|
|
|
|
server_ip = peewee.CharField(default="127.0.0.1")
|
|
|
|
server_port = peewee.IntegerField(default=25565)
|
|
|
|
logs_delete_after = peewee.IntegerField(default=0)
|
|
|
|
type = peewee.CharField(default="minecraft-java")
|
|
|
|
show_status = peewee.BooleanField(default=1)
|
|
|
|
created_by = peewee.IntegerField(default=-100)
|
|
|
|
shutdown_timeout = peewee.IntegerField(default=60)
|
|
|
|
ignored_exits = peewee.CharField(default="0")
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
table_name = "servers"
|
|
|
|
database = db
|
|
|
|
|
2024-06-15 14:32:18 -04:00
|
|
|
class Backups(peewee.Model):
|
|
|
|
backup_id = peewee.CharField(primary_key=True, default=Helpers.create_uuid)
|
|
|
|
backup_name = peewee.CharField(default="New Backup")
|
|
|
|
backup_location = peewee.CharField(default="")
|
|
|
|
excluded_dirs = peewee.CharField(null=True)
|
|
|
|
max_backups = peewee.IntegerField()
|
|
|
|
server_id = peewee.ForeignKeyField(Servers, backref="backups_server")
|
|
|
|
compress = peewee.BooleanField(default=False)
|
|
|
|
shutdown = peewee.BooleanField(default=False)
|
|
|
|
before = peewee.CharField(default="")
|
|
|
|
after = peewee.CharField(default="")
|
|
|
|
default = peewee.BooleanField(default=False)
|
|
|
|
status = peewee.CharField(default='{"status": "Standby", "message": ""}')
|
|
|
|
enabled = peewee.BooleanField(default=True)
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
table_name = "backups"
|
|
|
|
database = db
|
|
|
|
|
2024-03-08 22:06:33 -05:00
|
|
|
class NewBackups(peewee.Model):
|
2024-06-02 10:47:13 -04:00
|
|
|
backup_id = peewee.CharField(primary_key=True, default=Helpers.create_uuid)
|
2024-03-08 22:06:33 -05:00
|
|
|
backup_name = peewee.CharField(default="New Backup")
|
|
|
|
backup_location = peewee.CharField(default="")
|
|
|
|
excluded_dirs = peewee.CharField(null=True)
|
|
|
|
max_backups = peewee.IntegerField()
|
|
|
|
server_id = peewee.ForeignKeyField(Servers, backref="backups_server")
|
|
|
|
compress = peewee.BooleanField(default=False)
|
|
|
|
shutdown = peewee.BooleanField(default=False)
|
|
|
|
before = peewee.CharField(default="")
|
|
|
|
after = peewee.CharField(default="")
|
2024-05-26 13:45:13 -04:00
|
|
|
default = peewee.BooleanField(default=False)
|
|
|
|
status = peewee.CharField(default='{"status": "Standby", "message": ""}')
|
2024-05-25 13:51:40 -04:00
|
|
|
enabled = peewee.BooleanField(default=True)
|
2024-03-08 22:06:33 -05:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
table_name = "new_backups"
|
|
|
|
database = db
|
|
|
|
|
2024-06-15 14:32:18 -04:00
|
|
|
class Schedules(peewee.Model):
|
|
|
|
schedule_id = peewee.IntegerField(unique=True, primary_key=True)
|
|
|
|
server_id = peewee.ForeignKeyField(Servers, backref="schedule_server")
|
|
|
|
enabled = peewee.BooleanField()
|
|
|
|
action = peewee.CharField()
|
|
|
|
interval = peewee.IntegerField()
|
|
|
|
interval_type = peewee.CharField()
|
|
|
|
start_time = peewee.CharField(null=True)
|
|
|
|
command = peewee.CharField(null=True)
|
|
|
|
action_id = peewee.CharField(null=True)
|
|
|
|
name = peewee.CharField()
|
|
|
|
one_time = peewee.BooleanField(default=False)
|
|
|
|
cron_string = peewee.CharField(default="")
|
|
|
|
parent = peewee.IntegerField(null=True)
|
|
|
|
delay = peewee.IntegerField(default=0)
|
|
|
|
next_run = peewee.CharField(default="")
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
table_name = "schedules"
|
|
|
|
database = db
|
|
|
|
|
2024-05-25 13:51:40 -04:00
|
|
|
class NewSchedules(peewee.Model):
|
|
|
|
schedule_id = peewee.IntegerField(unique=True, primary_key=True)
|
|
|
|
server_id = peewee.ForeignKeyField(Servers, backref="schedule_server")
|
|
|
|
enabled = peewee.BooleanField()
|
|
|
|
action = peewee.CharField()
|
|
|
|
interval = peewee.IntegerField()
|
|
|
|
interval_type = peewee.CharField()
|
|
|
|
start_time = peewee.CharField(null=True)
|
|
|
|
command = peewee.CharField(null=True)
|
|
|
|
action_id = peewee.CharField(null=True)
|
|
|
|
name = peewee.CharField()
|
|
|
|
one_time = peewee.BooleanField(default=False)
|
|
|
|
cron_string = peewee.CharField(default="")
|
|
|
|
parent = peewee.IntegerField(null=True)
|
|
|
|
delay = peewee.IntegerField(default=0)
|
|
|
|
next_run = peewee.CharField(default="")
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
table_name = "new_schedules"
|
2024-06-15 14:32:18 -04:00
|
|
|
database = db
|
2024-05-25 13:51:40 -04:00
|
|
|
|
2024-03-08 22:06:33 -05:00
|
|
|
migrator.create_table(NewBackups)
|
2024-05-25 13:51:40 -04:00
|
|
|
migrator.create_table(NewSchedules)
|
2024-03-08 22:06:33 -05:00
|
|
|
|
|
|
|
migrator.run()
|
2024-07-29 18:04:01 -04:00
|
|
|
all_servers = [
|
|
|
|
row.server_id for row in Servers.select(Servers.server_id).distinct()
|
|
|
|
]
|
|
|
|
all_backups = Backups.select()
|
2024-08-13 00:12:22 -04:00
|
|
|
all_schedules = Schedules.select()
|
2024-07-29 18:04:01 -04:00
|
|
|
Console.info("Cleaning up orphan backups for all servers")
|
|
|
|
valid_backups = [
|
2024-08-13 00:12:22 -04:00
|
|
|
backup for backup in all_backups if is_valid_entry(backup, all_servers)
|
|
|
|
]
|
2024-09-16 14:54:43 -04:00
|
|
|
if len(valid_backups) < len(all_backups):
|
|
|
|
backup_migration_status = False
|
|
|
|
print("Orphan backup found")
|
2024-08-13 00:12:22 -04:00
|
|
|
Console.info("Cleaning up orphan schedules for all servers")
|
|
|
|
valid_schedules = [
|
|
|
|
schedule for schedule in all_schedules if is_valid_entry(schedule, all_servers)
|
2024-07-29 18:04:01 -04:00
|
|
|
]
|
2024-09-17 13:47:55 -04:00
|
|
|
if len(valid_schedules) < len(all_schedules):
|
2024-09-16 14:54:43 -04:00
|
|
|
schedule_migration_status = False
|
2024-03-08 22:06:33 -05:00
|
|
|
# Copy data from the existing backups table to the new one
|
2024-07-29 18:04:01 -04:00
|
|
|
for backup in valid_backups:
|
|
|
|
Console.info(f"Trying to get server for backup migration {backup.server_id}")
|
2024-03-08 22:06:33 -05:00
|
|
|
# Fetch the related server entry from the Servers table
|
|
|
|
server = Servers.get(Servers.server_id == backup.server_id)
|
2024-05-26 13:45:13 -04:00
|
|
|
Console.info(f"Migrations: Migrating backup for server {server.server_name}")
|
2024-05-25 15:11:46 -04:00
|
|
|
# Create a new backup entry with data from the
|
|
|
|
# old backup entry and related server
|
2024-06-07 14:08:05 -04:00
|
|
|
new_backup = NewBackups.create(
|
2024-05-26 13:45:13 -04:00
|
|
|
backup_name=f"{server.server_name} Backup",
|
2024-05-25 15:11:46 -04:00
|
|
|
# Set backup_location equal to backup_path
|
|
|
|
backup_location=server.backup_path,
|
2024-03-08 22:06:33 -05:00
|
|
|
excluded_dirs=backup.excluded_dirs,
|
|
|
|
max_backups=backup.max_backups,
|
|
|
|
server_id=server.server_id,
|
|
|
|
compress=backup.compress,
|
|
|
|
shutdown=backup.shutdown,
|
|
|
|
before=backup.before,
|
|
|
|
after=backup.after,
|
2024-05-26 13:45:13 -04:00
|
|
|
default=True,
|
2024-05-25 13:51:40 -04:00
|
|
|
enabled=True,
|
2024-03-08 22:06:33 -05:00
|
|
|
)
|
2024-07-29 18:04:01 -04:00
|
|
|
Console.info(
|
|
|
|
f"New backup table created for {server.server_name} with id {new_backup.backup_id}"
|
|
|
|
)
|
2024-06-07 14:08:05 -04:00
|
|
|
Helpers.ensure_dir_exists(
|
|
|
|
os.path.join(server.backup_path, new_backup.backup_id)
|
|
|
|
)
|
2024-07-29 18:04:01 -04:00
|
|
|
try:
|
|
|
|
Console.info(
|
|
|
|
f"Moving old backups to new backup dir for {server.server_name}"
|
|
|
|
)
|
|
|
|
for file in os.listdir(server.backup_path):
|
|
|
|
if not os.path.isdir(
|
|
|
|
os.path.join(os.path.join(server.backup_path, file))
|
|
|
|
):
|
|
|
|
FileHelpers.move_file(
|
|
|
|
os.path.join(server.backup_path, file),
|
|
|
|
os.path.join(server.backup_path, new_backup.backup_id, file),
|
|
|
|
)
|
|
|
|
except FileNotFoundError as why:
|
|
|
|
logger.error(
|
2024-08-06 18:36:21 -04:00
|
|
|
f"Could not move backups for {server.server_name} to new location with error {why}"
|
2024-07-29 18:04:01 -04:00
|
|
|
)
|
2024-03-08 22:06:33 -05:00
|
|
|
|
2024-05-26 13:45:13 -04:00
|
|
|
Console.debug("Migrations: Dropping old backup table")
|
2024-03-08 22:06:33 -05:00
|
|
|
# Drop the existing backups table
|
|
|
|
migrator.drop_table("backups")
|
|
|
|
|
2024-05-26 13:45:13 -04:00
|
|
|
Console.debug("Migrations: Renaming new_backups to backups")
|
2024-03-08 22:06:33 -05:00
|
|
|
# Rename the new table to backups
|
|
|
|
migrator.rename_table("new_backups", "backups")
|
2024-05-26 13:45:13 -04:00
|
|
|
|
|
|
|
Console.debug("Migrations: Dropping backup_path from servers table")
|
2024-03-08 22:06:33 -05:00
|
|
|
migrator.drop_columns("servers", ["backup_path"])
|
|
|
|
|
2024-08-13 00:12:22 -04:00
|
|
|
for schedule in valid_schedules:
|
2024-05-25 13:51:40 -04:00
|
|
|
action_id = None
|
|
|
|
if schedule.command == "backup_server":
|
2024-05-26 13:45:13 -04:00
|
|
|
Console.info(
|
|
|
|
f"Migrations: Adding backup ID to task with name {schedule.name}"
|
|
|
|
)
|
2024-08-13 00:12:22 -04:00
|
|
|
try:
|
|
|
|
backup = NewBackups.get(NewBackups.server_id == schedule.server_id)
|
|
|
|
except:
|
2024-09-16 17:37:04 -04:00
|
|
|
schedule_migration_status = False
|
2024-08-13 00:12:22 -04:00
|
|
|
Console.error(
|
|
|
|
"Could not find backup with selected server ID. Omitting from register."
|
|
|
|
)
|
|
|
|
continue
|
2024-05-25 13:51:40 -04:00
|
|
|
action_id = backup.backup_id
|
|
|
|
NewSchedules.create(
|
|
|
|
schedule_id=schedule.schedule_id,
|
|
|
|
server_id=schedule.server_id,
|
|
|
|
enabled=schedule.enabled,
|
|
|
|
action=schedule.action,
|
|
|
|
interval=schedule.interval,
|
|
|
|
interval_type=schedule.interval_type,
|
|
|
|
start_time=schedule.start_time,
|
|
|
|
command=schedule.command,
|
|
|
|
action_id=action_id,
|
|
|
|
name=schedule.name,
|
|
|
|
one_time=schedule.one_time,
|
|
|
|
cron_string=schedule.cron_string,
|
|
|
|
parent=schedule.parent,
|
|
|
|
delay=schedule.delay,
|
|
|
|
next_run=schedule.next_run,
|
|
|
|
)
|
|
|
|
|
2024-05-26 13:45:13 -04:00
|
|
|
Console.debug("Migrations: dropping old schedules table")
|
2024-05-25 13:51:40 -04:00
|
|
|
# Drop the existing backups table
|
|
|
|
migrator.drop_table("schedules")
|
|
|
|
|
2024-05-26 13:45:13 -04:00
|
|
|
Console.debug("Migrations: renaming new_schedules to schedules")
|
2024-05-25 13:51:40 -04:00
|
|
|
# Rename the new table to backups
|
|
|
|
migrator.rename_table("new_schedules", "schedules")
|
|
|
|
|
2024-09-16 17:37:04 -04:00
|
|
|
with open(
|
|
|
|
os.path.join(
|
|
|
|
os.path.abspath(os.path.curdir),
|
|
|
|
"app",
|
|
|
|
"migrations",
|
|
|
|
"status",
|
|
|
|
"20240308_multi-backup.json",
|
|
|
|
),
|
|
|
|
"w",
|
|
|
|
encoding="utf-8",
|
|
|
|
) as file:
|
2024-09-16 14:54:43 -04:00
|
|
|
file.write(
|
|
|
|
json.dumps(
|
|
|
|
{
|
|
|
|
"backup_migration": {
|
2024-09-16 17:37:04 -04:00
|
|
|
"type": "backup",
|
2024-09-16 14:54:43 -04:00
|
|
|
"status": backup_migration_status,
|
2024-09-16 17:37:04 -04:00
|
|
|
"pid": str(uuid.uuid4()),
|
2024-09-16 14:54:43 -04:00
|
|
|
},
|
|
|
|
"schedule_migration": {
|
2024-09-16 17:37:04 -04:00
|
|
|
"type": "schedule",
|
2024-09-16 14:54:43 -04:00
|
|
|
"status": schedule_migration_status,
|
2024-09-16 17:37:04 -04:00
|
|
|
"pid": str(uuid.uuid4()),
|
2024-09-16 14:54:43 -04:00
|
|
|
},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
2024-03-08 22:06:33 -05:00
|
|
|
|
|
|
|
def rollback(migrator: Migrator, database, **kwargs):
|
|
|
|
"""
|
|
|
|
Write your rollback migrations here.
|
|
|
|
"""
|
|
|
|
db = database
|
|
|
|
|
|
|
|
migrator.drop_columns("backups", ["name", "backup_id", "backup_location"])
|
|
|
|
migrator.add_columns("servers", backup_path=peewee.CharField(default=""))
|