2022-09-07 11:49:42 +00:00
|
|
|
from django.db import models
|
2023-07-24 03:52:25 +00:00
|
|
|
from django.utils.translation import gettext_lazy as _
|
2022-09-07 11:49:42 +00:00
|
|
|
|
2023-01-16 11:02:09 +00:00
|
|
|
from accounts.const import (
|
|
|
|
AutomationTypes, SecretType, SecretStrategy, SSHKeyStrategy
|
|
|
|
)
|
2023-03-08 10:52:00 +00:00
|
|
|
from accounts.models import Account
|
|
|
|
from common.db import fields
|
|
|
|
from common.db.models import JMSBaseModel
|
2023-01-16 11:02:09 +00:00
|
|
|
from .base import AccountBaseAutomation
|
2022-09-07 11:49:42 +00:00
|
|
|
|
2022-12-27 07:27:33 +00:00
|
|
|
__all__ = ['ChangeSecretAutomation', 'ChangeSecretRecord', 'ChangeSecretMixin']
|
2022-10-13 09:47:29 +00:00
|
|
|
|
|
|
|
|
2022-12-27 07:27:33 +00:00
|
|
|
class ChangeSecretMixin(models.Model):
|
2022-10-19 09:05:21 +00:00
|
|
|
secret_type = models.CharField(
|
|
|
|
choices=SecretType.choices, max_length=16,
|
2022-11-08 09:54:51 +00:00
|
|
|
default=SecretType.PASSWORD, verbose_name=_('Secret type')
|
2022-10-19 09:05:21 +00:00
|
|
|
)
|
2023-02-20 15:50:33 +00:00
|
|
|
secret = fields.EncryptTextField(blank=True, null=True, verbose_name=_('Secret'))
|
2022-10-19 09:05:21 +00:00
|
|
|
secret_strategy = models.CharField(
|
|
|
|
choices=SecretStrategy.choices, max_length=16,
|
2022-10-21 10:19:09 +00:00
|
|
|
default=SecretStrategy.custom, verbose_name=_('Secret strategy')
|
2022-10-19 09:05:21 +00:00
|
|
|
)
|
2022-10-13 09:47:29 +00:00
|
|
|
password_rules = models.JSONField(default=dict, verbose_name=_('Password rules'))
|
2022-10-19 09:05:21 +00:00
|
|
|
ssh_key_change_strategy = models.CharField(
|
|
|
|
choices=SSHKeyStrategy.choices, max_length=16,
|
|
|
|
default=SSHKeyStrategy.add, verbose_name=_('SSH key change strategy')
|
|
|
|
)
|
2022-12-27 07:27:33 +00:00
|
|
|
|
2023-03-08 10:52:00 +00:00
|
|
|
get_all_assets: callable # get all assets
|
|
|
|
|
2022-12-27 07:27:33 +00:00
|
|
|
class Meta:
|
|
|
|
abstract = True
|
|
|
|
|
2023-03-08 10:52:00 +00:00
|
|
|
def create_nonlocal_accounts(self, usernames, asset):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def get_account_ids(self):
|
|
|
|
usernames = self.accounts
|
|
|
|
accounts = Account.objects.none()
|
|
|
|
for asset in self.get_all_assets():
|
|
|
|
self.create_nonlocal_accounts(usernames, asset)
|
|
|
|
accounts = accounts | asset.accounts.all()
|
|
|
|
account_ids = accounts.filter(
|
|
|
|
username__in=usernames, secret_type=self.secret_type
|
|
|
|
).values_list('id', flat=True)
|
|
|
|
return [str(_id) for _id in account_ids]
|
|
|
|
|
2023-01-16 11:02:09 +00:00
|
|
|
def to_attr_json(self):
|
|
|
|
attr_json = super().to_attr_json()
|
|
|
|
attr_json.update({
|
|
|
|
'secret': self.secret,
|
|
|
|
'secret_type': self.secret_type,
|
2023-03-08 10:52:00 +00:00
|
|
|
'accounts': self.get_account_ids(),
|
2023-01-16 11:02:09 +00:00
|
|
|
'password_rules': self.password_rules,
|
2023-03-08 10:52:00 +00:00
|
|
|
'secret_strategy': self.secret_strategy,
|
2023-01-16 11:02:09 +00:00
|
|
|
'ssh_key_change_strategy': self.ssh_key_change_strategy,
|
|
|
|
})
|
|
|
|
return attr_json
|
|
|
|
|
|
|
|
|
|
|
|
class ChangeSecretAutomation(ChangeSecretMixin, AccountBaseAutomation):
|
2022-11-08 09:54:51 +00:00
|
|
|
recipients = models.ManyToManyField('users.User', verbose_name=_("Recipient"), blank=True)
|
2022-09-07 11:49:42 +00:00
|
|
|
|
2022-10-12 10:08:57 +00:00
|
|
|
def save(self, *args, **kwargs):
|
2022-10-19 09:05:21 +00:00
|
|
|
self.type = AutomationTypes.change_secret
|
2022-10-12 10:08:57 +00:00
|
|
|
super().save(*args, **kwargs)
|
|
|
|
|
2022-09-07 11:49:42 +00:00
|
|
|
class Meta:
|
2022-10-19 03:39:11 +00:00
|
|
|
verbose_name = _("Change secret automation")
|
2022-09-07 11:49:42 +00:00
|
|
|
|
2022-10-19 09:05:21 +00:00
|
|
|
def to_attr_json(self):
|
|
|
|
attr_json = super().to_attr_json()
|
|
|
|
attr_json.update({
|
|
|
|
'recipients': {
|
|
|
|
str(recipient.id): (str(recipient), bool(recipient.secret_key))
|
|
|
|
for recipient in self.recipients.all()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return attr_json
|
|
|
|
|
2022-10-13 09:47:29 +00:00
|
|
|
|
|
|
|
class ChangeSecretRecord(JMSBaseModel):
|
2023-01-16 11:02:09 +00:00
|
|
|
execution = models.ForeignKey('accounts.AutomationExecution', on_delete=models.CASCADE)
|
2022-11-03 14:39:48 +00:00
|
|
|
asset = models.ForeignKey('assets.Asset', on_delete=models.CASCADE, null=True)
|
2023-01-16 11:02:09 +00:00
|
|
|
account = models.ForeignKey('accounts.Account', on_delete=models.CASCADE, null=True)
|
2022-10-13 09:47:29 +00:00
|
|
|
old_secret = fields.EncryptTextField(blank=True, null=True, verbose_name=_('Old secret'))
|
|
|
|
new_secret = fields.EncryptTextField(blank=True, null=True, verbose_name=_('Secret'))
|
|
|
|
date_started = models.DateTimeField(blank=True, null=True, verbose_name=_('Date started'))
|
|
|
|
date_finished = models.DateTimeField(blank=True, null=True, verbose_name=_('Date finished'))
|
|
|
|
status = models.CharField(max_length=16, default='pending')
|
|
|
|
error = models.TextField(blank=True, null=True, verbose_name=_('Error'))
|
|
|
|
|
|
|
|
class Meta:
|
2023-01-16 11:02:09 +00:00
|
|
|
ordering = ('-date_created',)
|
2022-10-19 09:05:21 +00:00
|
|
|
verbose_name = _("Change secret record")
|
2022-10-13 09:47:29 +00:00
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return self.account.__str__()
|
2022-11-21 07:18:09 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def timedelta(self):
|
|
|
|
if self.date_started and self.date_finished:
|
|
|
|
return self.date_finished - self.date_started
|
|
|
|
return None
|