mirror of https://github.com/jumpserver/jumpserver
237 lines
8.2 KiB
Python
237 lines
8.2 KiB
Python
|
import os
|
||
|
import time
|
||
|
from openpyxl import Workbook
|
||
|
from collections import defaultdict, OrderedDict
|
||
|
|
||
|
from django.conf import settings
|
||
|
from django.db.models import F
|
||
|
from rest_framework import serializers
|
||
|
|
||
|
from assets.models import Account
|
||
|
from assets.const import AllTypes
|
||
|
from assets.serializers import AccountSecretSerializer
|
||
|
from assets.notifications import AccountBackupExecutionTaskMsg
|
||
|
from users.models import User
|
||
|
from common.utils import get_logger
|
||
|
from common.utils.timezone import local_now_display
|
||
|
from common.utils.file import encrypt_and_compress_zip_file
|
||
|
|
||
|
logger = get_logger(__file__)
|
||
|
|
||
|
PATH = os.path.join(os.path.dirname(settings.BASE_DIR), 'tmp')
|
||
|
|
||
|
|
||
|
class BaseAccountHandler:
|
||
|
@classmethod
|
||
|
def unpack_data(cls, serializer_data, data=None):
|
||
|
if data is None:
|
||
|
data = {}
|
||
|
for k, v in serializer_data.items():
|
||
|
if isinstance(v, OrderedDict):
|
||
|
cls.unpack_data(v, data)
|
||
|
else:
|
||
|
data[k] = v
|
||
|
return data
|
||
|
|
||
|
@classmethod
|
||
|
def get_header_fields(cls, serializer: serializers.Serializer):
|
||
|
try:
|
||
|
backup_fields = getattr(serializer, 'Meta').fields_backup
|
||
|
except AttributeError:
|
||
|
backup_fields = serializer.fields.keys()
|
||
|
header_fields = {}
|
||
|
for field in backup_fields:
|
||
|
v = serializer.fields[field]
|
||
|
if isinstance(v, serializers.Serializer):
|
||
|
_fields = cls.get_header_fields(v)
|
||
|
header_fields.update(_fields)
|
||
|
else:
|
||
|
header_fields[field] = str(v.label)
|
||
|
return header_fields
|
||
|
|
||
|
@staticmethod
|
||
|
def load_auth(tp, value, system_user):
|
||
|
if value:
|
||
|
return value
|
||
|
if system_user:
|
||
|
return getattr(system_user, tp, '')
|
||
|
return ''
|
||
|
|
||
|
@classmethod
|
||
|
def replace_auth(cls, account, system_user_dict):
|
||
|
system_user = system_user_dict.get(account.systemuser_id)
|
||
|
account.username = cls.load_auth('username', account.username, system_user)
|
||
|
account.password = cls.load_auth('password', account.password, system_user)
|
||
|
account.private_key = cls.load_auth('private_key', account.private_key, system_user)
|
||
|
account.public_key = cls.load_auth('public_key', account.public_key, system_user)
|
||
|
return account
|
||
|
|
||
|
@classmethod
|
||
|
def create_row(cls, data, header_fields):
|
||
|
data = cls.unpack_data(data)
|
||
|
row_dict = {}
|
||
|
for field, header_name in header_fields.items():
|
||
|
row_dict[header_name] = str(data.get(field, field))
|
||
|
return row_dict
|
||
|
|
||
|
@classmethod
|
||
|
def add_rows(cls, data, header_fields, sheet):
|
||
|
data_map = defaultdict(list)
|
||
|
for i in data:
|
||
|
row = cls.create_row(i, header_fields)
|
||
|
if sheet not in data_map:
|
||
|
data_map[sheet].append(list(row.keys()))
|
||
|
data_map[sheet].append(list(row.values()))
|
||
|
return data_map
|
||
|
|
||
|
|
||
|
class AssetAccountHandler(BaseAccountHandler):
|
||
|
@staticmethod
|
||
|
def get_filename(plan_name):
|
||
|
filename = os.path.join(
|
||
|
PATH, f'{plan_name}-{local_now_display()}-{time.time()}.xlsx'
|
||
|
)
|
||
|
return filename
|
||
|
|
||
|
@classmethod
|
||
|
def create_data_map(cls, categories: list):
|
||
|
data_map = defaultdict(list)
|
||
|
|
||
|
# TODO 可以优化一下查询 在账号上做 category 的缓存 避免数据量大时连表操作
|
||
|
qs = Account.objects.filter(
|
||
|
asset__platform__type__in=categories
|
||
|
).annotate(category=F('asset__platform__type'))
|
||
|
print(qs, categories)
|
||
|
if not qs.exists():
|
||
|
return data_map
|
||
|
|
||
|
category_dict = {}
|
||
|
for i in AllTypes.grouped_choices_to_objs():
|
||
|
for j in i['children']:
|
||
|
category_dict[j['value']] = j['display_name']
|
||
|
|
||
|
header_fields = cls.get_header_fields(AccountSecretSerializer(qs.first()))
|
||
|
account_category_map = defaultdict(list)
|
||
|
for account in qs:
|
||
|
account_category_map[account.category].append(account)
|
||
|
|
||
|
data_map = {}
|
||
|
for category, accounts in account_category_map.items():
|
||
|
sheet_name = category_dict.get(category, category)
|
||
|
data = AccountSecretSerializer(accounts, many=True).data
|
||
|
data_map.update(cls.add_rows(data, header_fields, sheet_name))
|
||
|
|
||
|
logger.info('\n\033[33m- 共收集 {} 条账号\033[0m'.format(qs.count()))
|
||
|
return data_map
|
||
|
|
||
|
|
||
|
class AccountBackupHandler:
|
||
|
def __init__(self, execution):
|
||
|
self.execution = execution
|
||
|
self.plan_name = self.execution.plan.name
|
||
|
self.is_frozen = False # 任务状态冻结标志
|
||
|
|
||
|
def create_excel(self):
|
||
|
logger.info(
|
||
|
'\n'
|
||
|
'\033[32m>>> 正在生成资产或应用相关备份信息文件\033[0m'
|
||
|
''
|
||
|
)
|
||
|
# Print task start date
|
||
|
time_start = time.time()
|
||
|
files = []
|
||
|
categories = self.execution.categories
|
||
|
|
||
|
data_map = AssetAccountHandler.create_data_map(categories)
|
||
|
if not data_map:
|
||
|
return files
|
||
|
|
||
|
filename = AssetAccountHandler.get_filename(self.plan_name)
|
||
|
|
||
|
wb = Workbook(filename)
|
||
|
for sheet, data in data_map.items():
|
||
|
ws = wb.create_sheet(str(sheet))
|
||
|
for row in data:
|
||
|
ws.append(row)
|
||
|
wb.save(filename)
|
||
|
files.append(filename)
|
||
|
timedelta = round((time.time() - time_start), 2)
|
||
|
logger.info('步骤完成: 用时 {}s'.format(timedelta))
|
||
|
return files
|
||
|
|
||
|
def send_backup_mail(self, files, recipients):
|
||
|
if not files:
|
||
|
return
|
||
|
recipients = User.objects.filter(id__in=list(recipients))
|
||
|
logger.info(
|
||
|
'\n'
|
||
|
'\033[32m>>> 发送备份邮件\033[0m'
|
||
|
''
|
||
|
)
|
||
|
plan_name = self.plan_name
|
||
|
for user in recipients:
|
||
|
if not user.secret_key:
|
||
|
attachment_list = []
|
||
|
else:
|
||
|
password = user.secret_key.encode('utf8')
|
||
|
attachment = os.path.join(PATH, f'{plan_name}-{local_now_display()}-{time.time()}.zip')
|
||
|
encrypt_and_compress_zip_file(attachment, password, files)
|
||
|
attachment_list = [attachment, ]
|
||
|
AccountBackupExecutionTaskMsg(plan_name, user).publish(attachment_list)
|
||
|
logger.info('邮件已发送至{}({})'.format(user, user.email))
|
||
|
for file in files:
|
||
|
os.remove(file)
|
||
|
|
||
|
def step_perform_task_update(self, is_success, reason):
|
||
|
self.execution.reason = reason[:1024]
|
||
|
self.execution.is_success = is_success
|
||
|
self.execution.save()
|
||
|
logger.info('已完成对任务状态的更新')
|
||
|
|
||
|
def step_finished(self, is_success):
|
||
|
if is_success:
|
||
|
logger.info('任务执行成功')
|
||
|
else:
|
||
|
logger.error('任务执行失败')
|
||
|
|
||
|
def _run(self):
|
||
|
is_success = False
|
||
|
error = '-'
|
||
|
try:
|
||
|
recipients = self.execution.plan_snapshot.get('recipients')
|
||
|
if not recipients:
|
||
|
logger.info(
|
||
|
'\n'
|
||
|
'\033[32m>>> 该备份任务未分配收件人\033[0m'
|
||
|
''
|
||
|
)
|
||
|
else:
|
||
|
files = self.create_excel()
|
||
|
self.send_backup_mail(files, recipients)
|
||
|
except Exception as e:
|
||
|
self.is_frozen = True
|
||
|
logger.error('任务执行被异常中断')
|
||
|
logger.info('下面打印发生异常的 Traceback 信息 : ')
|
||
|
logger.error(e, exc_info=True)
|
||
|
error = str(e)
|
||
|
else:
|
||
|
is_success = True
|
||
|
finally:
|
||
|
reason = error
|
||
|
self.step_perform_task_update(is_success, reason)
|
||
|
self.step_finished(is_success)
|
||
|
|
||
|
def run(self):
|
||
|
logger.info('任务开始: {}'.format(local_now_display()))
|
||
|
time_start = time.time()
|
||
|
try:
|
||
|
self._run()
|
||
|
except Exception as e:
|
||
|
logger.error('任务运行出现异常')
|
||
|
logger.error('下面显示异常 Traceback 信息: ')
|
||
|
logger.error(e, exc_info=True)
|
||
|
finally:
|
||
|
logger.info('\n任务结束: {}'.format(local_now_display()))
|
||
|
timedelta = round((time.time() - time_start), 2)
|
||
|
logger.info('用时: {}'.format(timedelta))
|