eduardogsilva.routerfleet/routerlib/backup_functions.py

185 lines
8.7 KiB
Python
Raw Normal View History

2024-03-20 16:59:34 -03:00
import datetime
from django.utils import timezone
from backup_data.models import RouterBackup
import os
from scp import SCPClient
from django.core.files.base import ContentFile
2024-04-10 13:33:10 -03:00
from routerlib.functions import gen_backup_name, connect_to_ssh, get_router_backup_file_extension
2024-03-20 16:59:34 -03:00
def perform_backup(router_backup: RouterBackup):
if router_backup.success or router_backup.error:
2024-04-03 11:12:11 -03:00
router_backup.router.routerstatus.backup_lock = None
router_backup.router.routerstatus.save()
2024-03-20 16:59:34 -03:00
return
if not router_backup.router.backup_profile:
router_backup.error = True
router_backup.error_message = "No backup profile assigned"
router_backup.save()
2024-04-03 11:12:11 -03:00
router_backup.router.routerstatus.backup_lock = None
router_backup.router.routerstatus.save()
2024-03-20 16:59:34 -03:00
return
if router_backup.retry_count > router_backup.router.backup_profile.max_retry:
router_backup.error = True
router_backup.save()
2024-04-03 11:12:11 -03:00
router_backup.router.routerstatus.backup_lock = None
router_backup.router.routerstatus.save()
2024-03-20 16:59:34 -03:00
return
if router_backup.backup_pending_retrieval:
backup_success, error_message = retrieve_backup(router_backup)
if error_message:
handle_backup_failure(router_backup, error_message)
return
if backup_success:
clean_up_backup_files(router_backup)
if router_backup.schedule_time:
start_time = router_backup.schedule_time
else:
start_time = router_backup.created
router_backup.queue_length = (timezone.now() - start_time).seconds
router_backup.finish_time = timezone.now()
router_backup.backup_pending_retrieval = False
router_backup.error_message = ''
router_backup.success = True
router_backup.save()
2024-04-03 12:54:59 -03:00
router_backup.router.routerstatus.last_backup = timezone.now()
router_backup.router.routerstatus.last_backup_failed = None
2024-04-03 11:12:11 -03:00
router_backup.router.routerstatus.backup_lock = None
router_backup.router.routerstatus.save()
2024-03-20 16:59:34 -03:00
else:
handle_backup_failure(router_backup, error_message)
else:
backup_success, backup_files, error_message = execute_backup(router_backup)
if backup_success:
router_backup.backup_pending_retrieval = True
router_backup.error_message = ''
router_backup.retry_count = 0
2024-04-02 16:56:35 -03:00
router_backup.next_retry = timezone.now() + datetime.timedelta(seconds=router_backup.router.backup_profile.retrieve_interval)
2024-03-20 16:59:34 -03:00
router_backup.save()
else:
handle_backup_failure(router_backup, error_message)
return
def handle_backup_failure(router_backup: RouterBackup, error_message):
router_backup.error_message = error_message
router_backup.retry_count += 1
router_backup.next_retry = timezone.now() + datetime.timedelta(minutes=router_backup.router.backup_profile.retry_interval)
router_backup.save()
def execute_backup(router_backup: RouterBackup):
error_message = ""
2024-04-10 09:58:06 -03:00
router = router_backup.router
2024-04-10 13:33:10 -03:00
backup_name = gen_backup_name(router_backup)
file_extension = get_router_backup_file_extension(router.router_type)
ssh_client = None
2024-03-20 16:59:34 -03:00
try:
if router_backup.router.router_type == 'routeros':
ssh_client = connect_to_ssh(router.address, router.port, router.username, router.password, router.ssh_key)
2024-04-10 13:33:10 -03:00
ssh_client.exec_command(f'/system backup save name={backup_name}.{file_extension["binary"]}')
ssh_client.exec_command(f'/export file={backup_name}.{file_extension["text"]}')
return True, [f'{backup_name}.{file_extension["binary"]}', f'{backup_name}.{file_extension["text"]}'], error_message
elif router_backup.router.router_type == 'openwrt':
ssh_client = connect_to_ssh(router.address, router.port, router.username, router.password, router.ssh_key)
2024-04-10 13:33:10 -03:00
stdin, stdout, stderr = ssh_client.exec_command('uci export')
backup_text = stdout.read().decode('utf-8')
if backup_text:
router_backup.backup_text = backup_text
router_backup.backup_text_filename = f'{backup_name}.{file_extension["text"]}'
router_backup.save()
else:
return False, [], "Failed to execute backup: Empty backup text"
ssh_client.exec_command(f'sysupgrade --create-backup /tmp/{backup_name}.{file_extension["binary"]}')
return True, [f'/tmp/{backup_name}.{file_extension["binary"]}', f'{backup_name}.{file_extension["text"]}'], error_message
2024-03-20 16:59:34 -03:00
else:
error_message = f"Router type not supported: {router_backup.router.get_router_type_display()}"
return False, [], error_message
except Exception as e:
error_message = f"Failed to execute backup: {str(e)}"
return False, [], error_message
finally:
if ssh_client:
ssh_client.close()
2024-03-20 16:59:34 -03:00
def retrieve_backup(router_backup: RouterBackup):
error_message = ""
2024-04-10 09:58:06 -03:00
router = router_backup.router
2024-04-01 12:36:50 -03:00
backup_name = gen_backup_name(router_backup)
2024-03-20 16:59:34 -03:00
success = False
2024-04-10 13:33:10 -03:00
file_extension = get_router_backup_file_extension(router.router_type)
ssh_client = None
2024-03-20 16:59:34 -03:00
try:
if router_backup.router.router_type == 'routeros':
2024-04-10 13:33:10 -03:00
rsc_file_path = f'/tmp/{backup_name}.{file_extension["text"]}'
backup_file_path = f'/tmp/{backup_name}.{file_extension["binary"]}'
ssh_client = connect_to_ssh(router.address, router.port, router.username, router.password, router.ssh_key)
2024-03-20 16:59:34 -03:00
scp_client = SCPClient(ssh_client.get_transport())
2024-04-10 13:33:10 -03:00
scp_client.get(f'/{backup_name}.{file_extension["text"]}', rsc_file_path)
scp_client.get(f'/{backup_name}.{file_extension["binary"]}', backup_file_path)
2024-03-20 16:59:34 -03:00
with open(rsc_file_path, 'r') as rsc_file:
rsc_content = rsc_file.read()
rsc_content_cleaned = '\n'.join(
line for line in rsc_content.split('\n') if not line.strip().startswith('#'))
router_backup.backup_text = rsc_content_cleaned
2024-04-10 13:33:10 -03:00
router_backup.backup_text_filename = f'{backup_name}.{file_extension["text"]}'
2024-03-20 16:59:34 -03:00
with open(backup_file_path, 'rb') as backup_file:
2024-04-10 13:33:10 -03:00
router_backup.backup_binary.save(f"{backup_name}.{file_extension['binary']}", ContentFile(backup_file.read()))
2024-03-20 16:59:34 -03:00
router_backup.save()
os.remove(rsc_file_path)
os.remove(backup_file_path)
2024-04-10 13:33:10 -03:00
ssh_client.exec_command(f'/file remove "{backup_name}.{file_extension["text"]}"')
ssh_client.exec_command(f'/file remove "{backup_name}.{file_extension["binary"]}"')
2024-03-20 16:59:34 -03:00
success = True
2024-04-10 13:33:10 -03:00
elif router_backup.router.router_type == 'openwrt':
remote_backup_file_path = f'/tmp/{backup_name}.{file_extension["binary"]}'
local_backup_file_path = f'/tmp/{backup_name}.{file_extension["binary"]}'
ssh_client = connect_to_ssh(router.address, router.port, router.username, router.password, router.ssh_key)
2024-04-10 13:33:10 -03:00
scp_client = SCPClient(ssh_client.get_transport())
scp_client.get(remote_backup_file_path, local_backup_file_path)
with open(local_backup_file_path, 'rb') as backup_file:
router_backup.backup_binary.save(f"{backup_name}.{file_extension['binary']}", ContentFile(backup_file.read()))
router_backup.save()
os.remove(local_backup_file_path)
ssh_client.exec_command(f'rm {remote_backup_file_path}')
success = True
2024-03-20 16:59:34 -03:00
else:
error_message = f"Router type not supported: {router_backup.router.get_router_type_display()}"
return success, error_message
except Exception as e:
return success, f"Failed to retrieve backup files: {str(e)}"
finally:
if ssh_client:
ssh_client.close()
2024-03-20 16:59:34 -03:00
return success, error_message
def clean_up_backup_files(router_backup: RouterBackup):
2024-04-10 09:58:06 -03:00
router = router_backup.router
ssh_client = None
2024-03-20 16:59:34 -03:00
try:
if router_backup.router.router_type == 'routeros':
ssh_client = connect_to_ssh(router.address, router.port, router.username, router.password, router.ssh_key)
2024-04-01 12:36:50 -03:00
ssh_client.exec_command('file remove [find where name~"routerfleet-backup-"]')
2024-04-10 13:33:10 -03:00
elif router_backup.router.router_type == 'openwrt':
ssh_client = connect_to_ssh(router.address, router.port, router.username, router.password, router.ssh_key)
2024-04-10 13:33:10 -03:00
ssh_client.exec_command('rm /tmp/routerfleet-backup-*')
2024-03-20 16:59:34 -03:00
else:
print(f"Router type not supported: {router_backup.router.get_router_type_display()}")
except Exception as e:
print(f"Failed to clean up backup files: {str(e)}")
finally:
if ssh_client:
ssh_client.close()