Adds backup manager - removes from server.py

This commit is contained in:
amcmanu3 2024-08-11 13:26:19 -04:00
parent c1327adf6b
commit 2fc20ae5b1
4 changed files with 245 additions and 183 deletions

View File

@ -11,6 +11,7 @@ from app.classes.shared.file_helpers import FileHelpers
from app.classes.shared.singleton import Singleton from app.classes.shared.singleton import Singleton
from app.classes.shared.server import ServerInstance from app.classes.shared.server import ServerInstance
from app.classes.shared.console import Console from app.classes.shared.console import Console
from app.classes.shared.backup_mgr import BackupManager
from app.classes.shared.helpers import Helpers from app.classes.shared.helpers import Helpers
from app.classes.shared.main_models import DatabaseShortcuts from app.classes.shared.main_models import DatabaseShortcuts
@ -39,6 +40,9 @@ class ServersController(metaclass=Singleton):
self.stats = Stats(self.helper, self) self.stats = Stats(self.helper, self)
self.web_sock = WebSocketManager() self.web_sock = WebSocketManager()
self.server_subpage = {} self.server_subpage = {}
self.backups_mgr = BackupManager(
self.helper, self.file_helper, self.management_helper
)
# ********************************************************************************** # **********************************************************************************
# Generic Servers Methods # Generic Servers Methods
@ -212,6 +216,7 @@ class ServersController(metaclass=Singleton):
self.management_helper, self.management_helper,
self.stats, self.stats,
self.file_helper, self.file_helper,
self.backups_mgr,
), ),
} }

View File

@ -0,0 +1,197 @@
import os
import time
import datetime
import json
import logging
from zoneinfo import ZoneInfo
# TZLocal is set as a hidden import on win pipeline
from zoneinfo import ZoneInfoNotFoundError
from tzlocal import get_localzone
from app.classes.models.management import HelpersManagement
from app.classes.models.users import HelperUsers
from app.classes.models.server_permissions import PermissionsServers
from app.classes.shared.console import Console
from app.classes.shared.helpers import Helpers
from app.classes.shared.websocket_manager import WebSocketManager
logger = logging.getLogger(__name__)
class BackupManager:
def __init__(self, helper, file_helper, management_helper):
self.helper = helper
self.file_helper = file_helper
self.management_helper = management_helper
try:
self.tz = get_localzone()
except ZoneInfoNotFoundError as e:
logger.error(
"Could not capture time zone from system. Falling back to Europe/London"
f" error: {e}"
)
self.tz = ZoneInfo("Europe/London")
def backup_starter(self, backup_config, server):
if backup_config.get("type", "zip_vault") == "zip_vault":
self.zip_vault(backup_config, server)
def zip_vault(self, backup_config, server):
logger.info(f"Starting server {server.name}" f" (ID {server.server_id}) backup")
server_users = PermissionsServers.get_server_user_list(server.server_id)
# Alert the start of the backup to the authorized users.
for user in server_users:
WebSocketManager().broadcast_user(
user,
"notification",
self.helper.translation.translate(
"notify", "backupStarted", HelperUsers.get_user_lang_by_id(user)
).format(server.name),
)
time.sleep(3)
# Adjust the location to include the backup ID for destination.
backup_location = os.path.join(
backup_config["backup_location"], backup_config["backup_id"]
)
# Check if the backup location even exists.
if not backup_location:
Console.critical("No backup path found. Canceling")
return None
self.helper.ensure_dir_exists(backup_location)
try:
backup_filename = (
f"{backup_location}/"
f"{datetime.datetime.now().astimezone(self.tz).strftime('%Y-%m-%d_%H-%M-%S')}" # pylint: disable=line-too-long
)
logger.info(
f"Creating backup of server {server.name}"
f" (ID#{server.server_id}, path={server.server_path}) "
f"at '{backup_filename}'"
)
excluded_dirs = HelpersManagement.get_excluded_backup_dirs(
backup_config["backup_id"]
)
server_dir = Helpers.get_os_understandable_path(server.server_path)
self.file_helper.make_backup(
Helpers.get_os_understandable_path(backup_filename),
server_dir,
excluded_dirs,
server.server_id,
backup_config["backup_id"],
backup_config["backup_name"],
backup_config["compress"],
)
self.remove_old_backups(backup_config, server)
logger.info(f"Backup of server: {server.name} completed")
results = {
"percent": 100,
"total_files": 0,
"current_file": 0,
"backup_id": backup_config["backup_id"],
}
if len(WebSocketManager().clients) > 0:
WebSocketManager().broadcast_page_params(
"/panel/server_detail",
{"id": str(server.server_id)},
"backup_status",
results,
)
server_users = PermissionsServers.get_server_user_list(server.server_id)
for user in server_users:
WebSocketManager().broadcast_user(
user,
"notification",
self.helper.translation.translate(
"notify",
"backupComplete",
HelperUsers.get_user_lang_by_id(user),
).format(server.name),
)
# pause to let people read message.
HelpersManagement.update_backup_config(
backup_config["backup_id"],
{"status": json.dumps({"status": "Standby", "message": ""})},
)
time.sleep(5)
except Exception as e:
logger.exception(
"Failed to create backup of server"
f" {server.name} (ID {server.server_id})"
)
results = {
"percent": 100,
"total_files": 0,
"current_file": 0,
"backup_id": backup_config["backup_id"],
}
if len(WebSocketManager().clients) > 0:
WebSocketManager().broadcast_page_params(
"/panel/server_detail",
{"id": str(server.server_id)},
"backup_status",
results,
)
HelpersManagement.update_backup_config(
backup_config["backup_id"],
{"status": json.dumps({"status": "Failed", "message": f"{e}"})},
)
server.backup_server(
backup_config,
)
def list_backups(self, backup_config: dict, server_id) -> list:
if not backup_config:
logger.info(
f"Error putting backup file list for server with ID: {server_id}"
)
return []
backup_location = os.path.join(
backup_config["backup_location"],
backup_config["backup_id"],
)
if not Helpers.check_path_exists(
Helpers.get_os_understandable_path(backup_location)
):
return []
files = Helpers.get_human_readable_files_sizes(
Helpers.list_dir_by_date(
Helpers.get_os_understandable_path(backup_location)
)
)
return [
{
"path": os.path.relpath(
f["path"],
start=Helpers.get_os_understandable_path(backup_location),
),
"size": f["size"],
}
for f in files
if f["path"].endswith(".zip")
]
def remove_old_backups(self, backup_config, server):
while (
len(self.list_backups(backup_config, server)) > backup_config["max_backups"]
and backup_config["max_backups"] > 0
):
backup_list = self.list_backups(backup_config, server.server_id)
oldfile = backup_list[0]
oldfile_path = os.path.join(
backup_config["backup_location"],
backup_config["backup_id"],
oldfile["path"],
)
logger.info(f"Removing old backup '{oldfile['path']}'")
os.remove(Helpers.get_os_understandable_path(oldfile_path))

View File

@ -31,6 +31,7 @@ from app.classes.models.server_stats import HelperServerStats
from app.classes.models.management import HelpersManagement, HelpersWebhooks from app.classes.models.management import HelpersManagement, HelpersWebhooks
from app.classes.models.users import HelperUsers from app.classes.models.users import HelperUsers
from app.classes.models.server_permissions import PermissionsServers from app.classes.models.server_permissions import PermissionsServers
from app.classes.shared.backup_mgr import BackupManager
from app.classes.shared.console import Console from app.classes.shared.console import Console
from app.classes.shared.helpers import Helpers from app.classes.shared.helpers import Helpers
from app.classes.shared.file_helpers import FileHelpers from app.classes.shared.file_helpers import FileHelpers
@ -160,10 +161,13 @@ class ServerInstance:
stats: Stats stats: Stats
stats_helper: HelperServerStats stats_helper: HelperServerStats
def __init__(self, server_id, helper, management_helper, stats, file_helper): def __init__(
self, server_id, helper, management_helper, stats, file_helper, backup_mgr
):
self.helper = helper self.helper = helper
self.file_helper = file_helper self.file_helper = file_helper
self.management_helper = management_helper self.management_helper = management_helper
self.backup_mgr = backup_mgr
# holders for our process # holders for our process
self.process = None self.process = None
self.line = False self.line = False
@ -1111,12 +1115,31 @@ class ServerInstance:
# Check to see if we're already backing up # Check to see if we're already backing up
if self.check_backup_by_id(backup_id): if self.check_backup_by_id(backup_id):
return False return False
backup_config = HelpersManagement.get_backup_config(backup_id)
if backup_config["before"]:
logger.debug(
"Found running server and send command option. Sending command"
)
self.send_command(backup_config["before"])
# Pause to let command run
time.sleep(5)
self.was_running = False
if backup_config["shutdown"]:
logger.info(
"Found shutdown preference. Delaying"
+ "backup start. Shutting down server."
)
if not update:
if self.check_running():
self.stop_server()
self.was_running = True
backup_thread = threading.Thread( backup_thread = threading.Thread(
target=self.backup_server, target=self.backup_mgr.backup_starter,
daemon=True, daemon=True,
name=f"backup_{backup_id}", name=f"backup_{backup_config['backup_id']}",
args=[backup_id, update], args=[backup_config, self],
) )
logger.info( logger.info(
f"Starting Backup Thread for server {self.settings['server_name']}." f"Starting Backup Thread for server {self.settings['server_name']}."
@ -1136,158 +1159,22 @@ class ServerInstance:
logger.info(f"Backup Thread started for server {self.settings['server_name']}.") logger.info(f"Backup Thread started for server {self.settings['server_name']}.")
@callback @callback
def backup_server(self, backup_id, update): def backup_server(self, backup_config):
was_server_running = None """
logger.info(f"Starting server {self.name} (ID {self.server_id}) backup") Called from backup manager as a backup finishes. The completion of this method
server_users = PermissionsServers.get_server_user_list(self.server_id) will send the webhook (that's why it's named backup_server).
# Alert the start of the backup to the authorized users. This will also call to set the backup status, turn the server back on, or
for user in server_users: send after commands.
WebSocketManager().broadcast_user( """
user, if backup_config["after"]:
"notification", self.send_command(backup_config["after"])
self.helper.translation.translate( if self.was_running:
"notify", "backupStarted", HelperUsers.get_user_lang_by_id(user)
).format(self.name),
)
time.sleep(3)
# Get the backup config
conf = HelpersManagement.get_backup_config(backup_id)
# Adjust the location to include the backup ID for destination.
backup_location = os.path.join(conf["backup_location"], conf["backup_id"])
# Check if the backup location even exists.
if not backup_location:
Console.critical("No backup path found. Canceling")
return None
if conf["before"]:
logger.debug(
"Found running server and send command option. Sending command"
)
self.send_command(conf["before"])
# Pause to let command run
time.sleep(5)
if conf["shutdown"]:
logger.info( logger.info(
"Found shutdown preference. Delaying" "Backup complete. User had shutdown preference. Starting server."
+ "backup start. Shutting down server."
)
if not update:
was_server_running = False
if self.check_running():
self.stop_server()
was_server_running = True
self.helper.ensure_dir_exists(backup_location)
try:
backup_filename = (
f"{backup_location}/"
f"{datetime.datetime.now().astimezone(self.tz).strftime('%Y-%m-%d_%H-%M-%S')}" # pylint: disable=line-too-long
)
logger.info(
f"Creating backup of server '{self.settings['server_name']}'"
f" (ID#{self.server_id}, path={self.server_path}) "
f"at '{backup_filename}'"
)
excluded_dirs = HelpersManagement.get_excluded_backup_dirs(backup_id)
server_dir = Helpers.get_os_understandable_path(self.settings["path"])
self.file_helper.make_backup(
Helpers.get_os_understandable_path(backup_filename),
server_dir,
excluded_dirs,
self.server_id,
backup_id,
conf["backup_name"],
conf["compress"],
)
while (
len(self.list_backups(conf)) > conf["max_backups"]
and conf["max_backups"] > 0
):
backup_list = self.list_backups(conf)
oldfile = backup_list[0]
oldfile_path = f"{backup_location}/{oldfile['path']}"
logger.info(f"Removing old backup '{oldfile['path']}'")
os.remove(Helpers.get_os_understandable_path(oldfile_path))
logger.info(f"Backup of server: {self.name} completed")
results = {
"percent": 100,
"total_files": 0,
"current_file": 0,
"backup_id": backup_id,
}
if len(WebSocketManager().clients) > 0:
WebSocketManager().broadcast_page_params(
"/panel/server_detail",
{"id": str(self.server_id)},
"backup_status",
results,
)
server_users = PermissionsServers.get_server_user_list(self.server_id)
for user in server_users:
WebSocketManager().broadcast_user(
user,
"notification",
self.helper.translation.translate(
"notify",
"backupComplete",
HelperUsers.get_user_lang_by_id(user),
).format(self.name),
)
if was_server_running:
logger.info(
"Backup complete. User had shutdown preference. Starting server."
)
self.run_threaded_server(HelperUsers.get_user_id_by_name("system"))
time.sleep(3)
if conf["after"]:
if self.check_running():
logger.debug(
"Found running server and send command option. Sending command"
)
self.send_command(conf["after"])
# pause to let people read message.
HelpersManagement.update_backup_config(
backup_id,
{"status": json.dumps({"status": "Standby", "message": ""})},
)
time.sleep(5)
except Exception as e:
logger.exception(
f"Failed to create backup of server {self.name} (ID {self.server_id})"
)
results = {
"percent": 100,
"total_files": 0,
"current_file": 0,
"backup_id": backup_id,
}
if len(WebSocketManager().clients) > 0:
WebSocketManager().broadcast_page_params(
"/panel/server_detail",
{"id": str(self.server_id)},
"backup_status",
results,
)
if was_server_running:
logger.info(
"Backup complete. User had shutdown preference. Starting server."
)
self.run_threaded_server(HelperUsers.get_user_id_by_name("system"))
HelpersManagement.update_backup_config(
backup_id,
{"status": json.dumps({"status": "Failed", "message": f"{e}"})},
) )
self.run_threaded_server(HelperUsers.get_user_id_by_name("system"))
self.set_backup_status() self.set_backup_status()
def last_backup_status(self):
return self.last_backup_failed
def set_backup_status(self): def set_backup_status(self):
backups = HelpersManagement.get_backups_by_server(self.server_id, True) backups = HelpersManagement.get_backups_by_server(self.server_id, True)
alert = False alert = False
@ -1296,35 +1183,8 @@ class ServerInstance:
alert = True alert = True
self.last_backup_failed = alert self.last_backup_failed = alert
def list_backups(self, backup_config: dict) -> list: def last_backup_status(self):
if not backup_config: return self.last_backup_failed
logger.info(
f"Error putting backup file list for server with ID: {self.server_id}"
)
return []
backup_location = os.path.join(
backup_config["backup_location"], backup_config["backup_id"]
)
if not Helpers.check_path_exists(
Helpers.get_os_understandable_path(backup_location)
):
return []
files = Helpers.get_human_readable_files_sizes(
Helpers.list_dir_by_date(
Helpers.get_os_understandable_path(backup_location)
)
)
return [
{
"path": os.path.relpath(
f["path"],
start=Helpers.get_os_understandable_path(backup_location),
),
"size": f["size"],
}
for f in files
if f["path"].endswith(".zip")
]
@callback @callback
def jar_update(self): def jar_update(self):

View File

@ -1291,8 +1291,8 @@ class PanelHandler(BaseHandler):
).is_backingup ).is_backingup
self.controller.servers.refresh_server_settings(server_id) self.controller.servers.refresh_server_settings(server_id)
try: try:
page_data["backup_list"] = server.list_backups( page_data["backup_list"] = server.backup_mgr.list_backups(
page_data["backup_config"] page_data["backup_config"], server.server_id
) )
except: except:
page_data["backup_list"] = [] page_data["backup_list"] = []