jumpserver/apps/terminal/models/applet/host.py

116 lines
4.3 KiB
Python
Raw Normal View History

2022-11-03 08:55:38 +00:00
import os
from collections import defaultdict
2022-10-22 03:17:02 +00:00
from django.db import models
from django.utils.translation import gettext_lazy as _
2022-11-03 08:55:38 +00:00
from django.utils import timezone
from rest_framework.exceptions import ValidationError
from simple_history.utils import bulk_create_with_history
2022-10-22 03:17:02 +00:00
2022-11-01 03:52:51 +00:00
from common.db.models import JMSBaseModel
2022-11-03 08:55:38 +00:00
from common.utils import random_string
2022-10-27 08:26:15 +00:00
from assets.models import Host
2022-10-22 03:17:02 +00:00
2022-11-01 03:52:51 +00:00
__all__ = ['AppletHost', 'AppletHostDeployment']
2022-10-22 03:17:02 +00:00
2022-10-27 08:26:15 +00:00
class AppletHost(Host):
2022-11-01 09:04:44 +00:00
deploy_options = models.JSONField(default=dict, verbose_name=_('Deploy options'))
2022-11-01 03:52:51 +00:00
inited = models.BooleanField(default=False, verbose_name=_('Inited'))
date_inited = models.DateTimeField(null=True, blank=True, verbose_name=_('Date inited'))
2022-10-22 03:17:02 +00:00
date_synced = models.DateTimeField(null=True, blank=True, verbose_name=_('Date synced'))
2022-11-02 12:33:27 +00:00
terminal = models.OneToOneField(
'terminal.Terminal', on_delete=models.PROTECT, null=True, blank=True,
related_name='applet_host', verbose_name=_('Terminal')
)
2022-10-22 03:17:02 +00:00
applets = models.ManyToManyField(
'Applet', verbose_name=_('Applet'),
2022-10-25 04:57:34 +00:00
through='AppletPublication', through_fields=('host', 'applet'),
2022-10-22 03:17:02 +00:00
)
2022-11-03 08:55:38 +00:00
LOCKING_ORG = '00000000-0000-0000-0000-000000000004'
2022-10-22 03:17:02 +00:00
2022-10-25 04:57:34 +00:00
def __str__(self):
2022-10-28 10:19:44 +00:00
return self.name
2022-10-25 04:57:34 +00:00
2022-11-03 08:55:38 +00:00
@property
def status(self):
if self.terminal:
return 'online'
return self.terminal.status
def check_terminal_binding(self, request):
request_terminal = getattr(request.user, 'terminal', None)
if not request_terminal:
raise ValidationError('Request user has no terminal')
self.date_synced = timezone.now()
if not self.terminal:
self.terminal = request_terminal
self.save(update_fields=['terminal', 'date_synced'])
elif self.terminal and self.terminal != request_terminal:
raise ValidationError('Terminal has been set')
else:
self.save(update_fields=['date_synced'])
def check_applets_state(self, applets_value_list):
applets = self.applets.all()
name_version_mapper = {
value['name']: value['version']
for value in applets_value_list
}
status_applets = defaultdict(list)
for applet in applets:
if applet.name not in name_version_mapper:
status_applets['unpublished'].append(applet)
elif applet.version != name_version_mapper[applet.name]:
status_applets['not_match'].append(applet)
else:
status_applets['published'].append(applet)
for status, applets in status_applets.items():
self.publications.filter(applet__in=applets)\
.exclude(status=status)\
.update(status=status)
@staticmethod
def random_username():
return 'jms_' + random_string(8)
@staticmethod
def random_password():
return random_string(16, special_char=True)
def generate_accounts(self):
amount = int(os.getenv('TERMINAL_ACCOUNTS_AMOUNT', 100))
now_count = self.accounts.filter(privileged=False).count()
need = amount - now_count
accounts = []
account_model = self.accounts.model
for i in range(need):
username = self.random_username()
password = self.random_password()
account = account_model(
username=username, secret=password, name=username,
asset_id=self.id, secret_type='password', version=1,
org_id=self.LOCKING_ORG
)
accounts.append(account)
bulk_create_with_history(accounts, account_model, batch_size=20)
2022-11-01 03:52:51 +00:00
class AppletHostDeployment(JMSBaseModel):
host = models.ForeignKey('AppletHost', on_delete=models.CASCADE, verbose_name=_('Hosting'))
2022-11-01 09:04:44 +00:00
initial = models.BooleanField(default=False, verbose_name=_('Initial'))
2022-11-01 03:52:51 +00:00
status = models.CharField(max_length=16, default='', verbose_name=_('Status'))
date_start = models.DateTimeField(null=True, verbose_name=_('Date start'), db_index=True)
date_finished = models.DateTimeField(null=True, verbose_name=_("Date finished"))
comment = models.TextField(default='', blank=True, verbose_name=_('Comment'))
def start(self, **kwargs):
from ...automations.deploy_applet_host import DeployAppletHostManager
manager = DeployAppletHostManager(self)
manager.run(**kwargs)