mirror of https://github.com/jumpserver/jumpserver
254 lines
9.3 KiB
Python
254 lines
9.3 KiB
Python
import os
|
|
import time
|
|
from copy import deepcopy
|
|
|
|
from django.conf import settings
|
|
from django.utils import timezone
|
|
from django.utils.translation import gettext_lazy as _
|
|
from xlsxwriter import Workbook
|
|
|
|
from accounts.const import AutomationTypes, SecretType, SSHKeyStrategy, SecretStrategy
|
|
from accounts.models import ChangeSecretRecord
|
|
from accounts.notifications import ChangeSecretExecutionTaskMsg
|
|
from accounts.serializers import ChangeSecretRecordBackUpSerializer
|
|
from assets.const import HostTypes
|
|
from common.utils import get_logger
|
|
from common.utils.file import encrypt_and_compress_zip_file
|
|
from common.utils.timezone import local_now_filename
|
|
from users.models import User
|
|
from ..base.manager import AccountBasePlaybookManager
|
|
from ...utils import SecretGenerator
|
|
|
|
logger = get_logger(__name__)
|
|
|
|
|
|
class ChangeSecretManager(AccountBasePlaybookManager):
|
|
ansible_account_prefer = ''
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
self.record_id = self.execution.snapshot.get('record_id')
|
|
self.secret_type = self.execution.snapshot.get('secret_type')
|
|
self.secret_strategy = self.execution.snapshot.get(
|
|
'secret_strategy', SecretStrategy.custom
|
|
)
|
|
self.ssh_key_change_strategy = self.execution.snapshot.get(
|
|
'ssh_key_change_strategy', SSHKeyStrategy.add
|
|
)
|
|
self.account_ids = self.execution.snapshot['accounts']
|
|
self.name_recorder_mapper = {} # 做个映射,方便后面处理
|
|
|
|
@classmethod
|
|
def method_type(cls):
|
|
return AutomationTypes.change_secret
|
|
|
|
def get_ssh_params(self, account, secret, secret_type):
|
|
kwargs = {}
|
|
if secret_type != SecretType.SSH_KEY:
|
|
return kwargs
|
|
kwargs['strategy'] = self.ssh_key_change_strategy
|
|
kwargs['exclusive'] = 'yes' if kwargs['strategy'] == SSHKeyStrategy.set else 'no'
|
|
|
|
if kwargs['strategy'] == SSHKeyStrategy.set_jms:
|
|
username = account.username
|
|
path = f'/{username}' if username == "root" else f'/home/{username}'
|
|
kwargs['dest'] = f'{path}/.ssh/authorized_keys'
|
|
kwargs['regexp'] = '.*{}$'.format(secret.split()[2].strip())
|
|
return kwargs
|
|
|
|
def secret_generator(self, secret_type):
|
|
return SecretGenerator(
|
|
self.secret_strategy, secret_type,
|
|
self.execution.snapshot.get('password_rules')
|
|
)
|
|
|
|
def get_secret(self, secret_type):
|
|
if self.secret_strategy == SecretStrategy.custom:
|
|
return self.execution.snapshot['secret']
|
|
else:
|
|
return self.secret_generator(secret_type).get_secret()
|
|
|
|
def get_accounts(self, privilege_account):
|
|
if not privilege_account:
|
|
print(f'not privilege account')
|
|
return []
|
|
|
|
asset = privilege_account.asset
|
|
accounts = asset.accounts.all()
|
|
accounts = accounts.filter(id__in=self.account_ids)
|
|
if self.secret_type:
|
|
accounts = accounts.filter(secret_type=self.secret_type)
|
|
|
|
if settings.CHANGE_AUTH_PLAN_SECURE_MODE_ENABLED:
|
|
accounts = accounts.filter(privileged=False).exclude(
|
|
username__in=['root', 'administrator', privilege_account.username]
|
|
)
|
|
return accounts
|
|
|
|
def host_callback(
|
|
self, host, asset=None, account=None,
|
|
automation=None, path_dir=None, **kwargs
|
|
):
|
|
host = super().host_callback(
|
|
host, asset=asset, account=account, automation=automation,
|
|
path_dir=path_dir, **kwargs
|
|
)
|
|
if host.get('error'):
|
|
return host
|
|
|
|
accounts = self.get_accounts(account)
|
|
if not accounts:
|
|
print('没有发现待处理的账号: %s 用户ID: %s 类型: %s' % (
|
|
asset.name, self.account_ids, self.secret_type
|
|
))
|
|
return []
|
|
|
|
records = []
|
|
inventory_hosts = []
|
|
if asset.type == HostTypes.WINDOWS and self.secret_type == SecretType.SSH_KEY:
|
|
print(f'Windows {asset} does not support ssh key push')
|
|
return inventory_hosts
|
|
|
|
host['ssh_params'] = {}
|
|
for account in accounts:
|
|
h = deepcopy(host)
|
|
secret_type = account.secret_type
|
|
h['name'] += '(' + account.username + ')'
|
|
if self.secret_type is None:
|
|
new_secret = account.secret
|
|
else:
|
|
new_secret = self.get_secret(secret_type)
|
|
|
|
if self.record_id is None:
|
|
recorder = ChangeSecretRecord(
|
|
asset=asset, account=account, execution=self.execution,
|
|
old_secret=account.secret, new_secret=new_secret,
|
|
)
|
|
records.append(recorder)
|
|
else:
|
|
recorder = ChangeSecretRecord.objects.get(id=self.record_id)
|
|
|
|
self.name_recorder_mapper[h['name']] = recorder
|
|
|
|
private_key_path = None
|
|
if secret_type == SecretType.SSH_KEY:
|
|
private_key_path = self.generate_private_key_path(new_secret, path_dir)
|
|
new_secret = self.generate_public_key(new_secret)
|
|
|
|
h['ssh_params'].update(self.get_ssh_params(account, new_secret, secret_type))
|
|
h['account'] = {
|
|
'name': account.name,
|
|
'username': account.username,
|
|
'secret_type': secret_type,
|
|
'secret': account.escape_jinja2_syntax(new_secret),
|
|
'private_key_path': private_key_path,
|
|
'become': account.get_ansible_become_auth(),
|
|
}
|
|
if asset.platform.type == 'oracle':
|
|
h['account']['mode'] = 'sysdba' if account.privileged else None
|
|
inventory_hosts.append(h)
|
|
ChangeSecretRecord.objects.bulk_create(records)
|
|
return inventory_hosts
|
|
|
|
def on_host_success(self, host, result):
|
|
recorder = self.name_recorder_mapper.get(host)
|
|
if not recorder:
|
|
return
|
|
recorder.status = 'success'
|
|
recorder.date_finished = timezone.now()
|
|
recorder.save()
|
|
account = recorder.account
|
|
if not account:
|
|
print("Account not found, deleted ?")
|
|
return
|
|
account.secret = recorder.new_secret
|
|
account.date_updated = timezone.now()
|
|
account.save(update_fields=['secret', 'date_updated'])
|
|
|
|
def on_host_error(self, host, error, result):
|
|
recorder = self.name_recorder_mapper.get(host)
|
|
if not recorder:
|
|
return
|
|
recorder.status = 'failed'
|
|
recorder.date_finished = timezone.now()
|
|
recorder.error = error
|
|
recorder.save()
|
|
|
|
def on_runner_failed(self, runner, e):
|
|
logger.error("Account error: ", e)
|
|
|
|
def check_secret(self):
|
|
if self.secret_strategy == SecretStrategy.custom \
|
|
and not self.execution.snapshot['secret']:
|
|
print('Custom secret is empty')
|
|
return False
|
|
return True
|
|
|
|
@staticmethod
|
|
def get_summary(recorders):
|
|
total, succeed, failed = 0, 0, 0
|
|
for recorder in recorders:
|
|
if recorder.status == 'success':
|
|
succeed += 1
|
|
else:
|
|
failed += 1
|
|
total += 1
|
|
|
|
summary = _('Success: %s, Failed: %s, Total: %s') % (succeed, failed, total)
|
|
return summary
|
|
|
|
def run(self, *args, **kwargs):
|
|
if self.secret_type and not self.check_secret():
|
|
return
|
|
super().run(*args, **kwargs)
|
|
recorders = list(self.name_recorder_mapper.values())
|
|
summary = self.get_summary(recorders)
|
|
print(summary, end='')
|
|
|
|
if self.record_id:
|
|
return
|
|
|
|
self.send_recorder_mail(recorders, summary)
|
|
|
|
def send_recorder_mail(self, recorders, summary):
|
|
recipients = self.execution.recipients
|
|
if not recorders or not recipients:
|
|
return
|
|
|
|
recipients = User.objects.filter(id__in=list(recipients.keys()))
|
|
|
|
name = self.execution.snapshot['name']
|
|
path = os.path.join(os.path.dirname(settings.BASE_DIR), 'tmp')
|
|
filename = os.path.join(path, f'{name}-{local_now_filename()}-{time.time()}.xlsx')
|
|
if not self.create_file(recorders, filename):
|
|
return
|
|
|
|
for user in recipients:
|
|
attachments = []
|
|
if user.secret_key:
|
|
password = user.secret_key.encode('utf8')
|
|
attachment = os.path.join(path, f'{name}-{local_now_filename()}-{time.time()}.zip')
|
|
encrypt_and_compress_zip_file(attachment, password, [filename])
|
|
attachments = [attachment]
|
|
ChangeSecretExecutionTaskMsg(name, user, summary).publish(attachments)
|
|
os.remove(filename)
|
|
|
|
@staticmethod
|
|
def create_file(recorders, filename):
|
|
serializer_cls = ChangeSecretRecordBackUpSerializer
|
|
serializer = serializer_cls(recorders, many=True)
|
|
|
|
header = [str(v.label) for v in serializer.child.fields.values()]
|
|
rows = [[str(i) for i in row.values()] for row in serializer.data]
|
|
if not rows:
|
|
return False
|
|
|
|
rows.insert(0, header)
|
|
wb = Workbook(filename)
|
|
ws = wb.add_worksheet('Sheet1')
|
|
for row_index, row_data in enumerate(rows):
|
|
for col_index, col_data in enumerate(row_data):
|
|
ws.write_string(row_index, col_index, col_data)
|
|
wb.close()
|
|
return True
|