jumpserver/apps/assets/models/asset.py

276 lines
9.6 KiB
Python
Raw Normal View History

2016-12-20 16:43:52 +00:00
#!/usr/bin/env python
# -*- coding: utf-8 -*-
#
2017-11-23 06:08:01 +00:00
import uuid
import logging
2018-05-28 07:00:06 +00:00
import random
2018-05-31 11:47:57 +00:00
from functools import reduce
from collections import defaultdict
2017-11-23 06:08:01 +00:00
2016-12-20 16:43:52 +00:00
from django.db import models
from django.db.models import Q
2016-12-20 16:43:52 +00:00
from django.utils.translation import ugettext_lazy as _
2017-12-10 16:29:25 +00:00
from django.core.cache import cache
2016-12-20 16:43:52 +00:00
2017-12-15 07:50:15 +00:00
from ..const import ASSET_ADMIN_CONN_CACHE_KEY
from .user import AdminUser, SystemUser
2018-10-10 07:37:20 +00:00
from orgs.mixins import OrgModelMixin, OrgManager
2016-12-20 16:43:52 +00:00
2017-03-24 06:48:18 +00:00
__all__ = ['Asset']
2016-12-20 16:43:52 +00:00
logger = logging.getLogger(__name__)
2018-01-09 15:33:14 +00:00
def default_cluster():
from .cluster import Cluster
name = "Default"
defaults = {"name": name}
cluster, created = Cluster.objects.get_or_create(
defaults=defaults, name=name
)
2018-01-10 04:03:35 +00:00
return cluster.id
2018-01-09 15:33:14 +00:00
def default_node():
try:
2018-02-09 07:24:44 +00:00
from .node import Node
2018-10-16 04:37:42 +00:00
root = Node.root()
return root
except:
return None
2018-07-13 07:05:46 +00:00
class AssetQuerySet(models.QuerySet):
def active(self):
return self.filter(is_active=True)
def valid(self):
return self.active()
2018-07-12 16:00:35 +00:00
class Asset(OrgModelMixin):
2017-03-15 16:19:47 +00:00
# Important
PLATFORM_CHOICES = (
('Linux', 'Linux'),
('Unix', 'Unix'),
('MacOS', 'MacOS'),
('BSD', 'BSD'),
('Windows', 'Windows'),
('Windows2016', 'Windows(2016)'),
('Other', 'Other'),
)
SSH_PROTOCOL = 'ssh'
RDP_PROTOCOL = 'rdp'
TELNET_PROTOCOL = 'telnet'
PROTOCOL_CHOICES = (
(SSH_PROTOCOL, 'ssh'),
(RDP_PROTOCOL, 'rdp'),
(TELNET_PROTOCOL, 'telnet (beta)'),
)
2017-11-23 06:08:01 +00:00
id = models.UUIDField(default=uuid.uuid4, primary_key=True)
2018-07-15 10:39:11 +00:00
ip = models.GenericIPAddressField(max_length=32, verbose_name=_('IP'), db_index=True)
hostname = models.CharField(max_length=128, verbose_name=_('Hostname'))
protocol = models.CharField(max_length=128, default=SSH_PROTOCOL, choices=PROTOCOL_CHOICES, verbose_name=_('Protocol'))
2016-12-20 16:43:52 +00:00
port = models.IntegerField(default=22, verbose_name=_('Port'))
2018-07-15 10:39:11 +00:00
platform = models.CharField(max_length=128, choices=PLATFORM_CHOICES, default='Linux', verbose_name=_('Platform'))
domain = models.ForeignKey("assets.Domain", null=True, blank=True,
related_name='assets', verbose_name=_("Domain"),
on_delete=models.SET_NULL)
nodes = models.ManyToManyField('assets.Node', default=default_node,
related_name='assets',
verbose_name=_("Nodes"))
2017-03-15 16:19:47 +00:00
is_active = models.BooleanField(default=True, verbose_name=_('Is active'))
# Auth
admin_user = models.ForeignKey('assets.AdminUser', on_delete=models.PROTECT,
null=True, verbose_name=_("Admin user"))
2017-03-15 16:19:47 +00:00
# Some information
2018-07-15 10:39:11 +00:00
public_ip = models.GenericIPAddressField(max_length=32, blank=True, null=True, verbose_name=_('Public IP'))
number = models.CharField(max_length=32, null=True, blank=True, verbose_name=_('Asset number'))
2017-03-15 16:19:47 +00:00
# Collect
vendor = models.CharField(max_length=64, null=True, blank=True,
verbose_name=_('Vendor'))
model = models.CharField(max_length=54, null=True, blank=True,
verbose_name=_('Model'))
sn = models.CharField(max_length=128, null=True, blank=True,
verbose_name=_('Serial number'))
cpu_model = models.CharField(max_length=64, null=True, blank=True,
verbose_name=_('CPU model'))
2017-03-15 16:19:47 +00:00
cpu_count = models.IntegerField(null=True, verbose_name=_('CPU count'))
cpu_cores = models.IntegerField(null=True, verbose_name=_('CPU cores'))
2018-08-15 04:00:47 +00:00
cpu_vcpus = models.IntegerField(null=True, verbose_name=_('CPU vcpus'))
memory = models.CharField(max_length=64, null=True, blank=True,
verbose_name=_('Memory'))
disk_total = models.CharField(max_length=1024, null=True, blank=True,
verbose_name=_('Disk total'))
disk_info = models.CharField(max_length=1024, null=True, blank=True,
verbose_name=_('Disk info'))
os = models.CharField(max_length=128, null=True, blank=True,
verbose_name=_('OS'))
os_version = models.CharField(max_length=16, null=True, blank=True,
verbose_name=_('OS version'))
os_arch = models.CharField(max_length=16, blank=True, null=True,
verbose_name=_('OS arch'))
hostname_raw = models.CharField(max_length=128, blank=True, null=True,
verbose_name=_('Hostname raw'))
labels = models.ManyToManyField('assets.Label', blank=True,
related_name='assets',
verbose_name=_("Labels"))
created_by = models.CharField(max_length=32, null=True, blank=True,
verbose_name=_('Created by'))
date_created = models.DateTimeField(auto_now_add=True, null=True,
blank=True,
verbose_name=_('Date created'))
comment = models.TextField(max_length=128, default='', blank=True,
verbose_name=_('Comment'))
2016-12-20 16:43:52 +00:00
objects = OrgManager.from_queryset(AssetQuerySet)()
2017-11-01 15:23:11 +00:00
def __str__(self):
return '{0.hostname}({0.ip})'.format(self)
2016-12-20 17:03:52 +00:00
2016-12-20 16:43:52 +00:00
@property
def is_valid(self):
warning = ''
if not self.is_active:
warning += ' inactive'
else:
return True, ''
return False, warning
def is_unixlike(self):
2018-07-25 02:22:32 +00:00
if self.platform not in ("Windows", "Windows2016"):
return True
else:
return False
2018-04-07 16:16:37 +00:00
def get_nodes(self):
from .node import Node
nodes = self.nodes.all() or [Node.root()]
return nodes
2018-04-07 16:16:37 +00:00
2018-05-31 11:47:57 +00:00
def get_all_nodes(self, flat=False):
nodes = []
2018-06-01 07:34:08 +00:00
for node in self.get_nodes():
2018-05-31 11:47:57 +00:00
_nodes = node.get_ancestor(with_self=True)
_nodes.append(_nodes)
if flat:
nodes = list(reduce(lambda x, y: set(x) | set(y), nodes))
return nodes
@classmethod
def get_queryset_by_fullname_list(cls, fullname_list):
org_fullname_map = defaultdict(list)
for fullname in fullname_list:
hostname, org = cls.split_fullname(fullname)
org_fullname_map[org].append(hostname)
filter_arg = Q()
for org, hosts in org_fullname_map.items():
if org.is_real():
filter_arg |= Q(hostname__in=hosts, org_id=org.id)
else:
filter_arg |= Q(Q(org_id__isnull=True) | Q(org_id=''), hostname__in=hosts)
return Asset.objects.filter(filter_arg)
2018-08-01 05:06:50 +00:00
@property
2017-12-12 04:19:45 +00:00
def hardware_info(self):
if self.cpu_count:
return '{} Core {} {}'.format(
2018-08-15 04:00:47 +00:00
self.cpu_vcpus or self.cpu_count * self.cpu_cores,
2017-12-12 04:19:45 +00:00
self.memory, self.disk_total
)
else:
return ''
@property
2017-12-12 04:19:45 +00:00
def is_connective(self):
if not self.is_unixlike():
return True
2017-12-15 07:50:15 +00:00
val = cache.get(ASSET_ADMIN_CONN_CACHE_KEY.format(self.hostname))
2017-12-12 04:19:45 +00:00
if val == 1:
return True
else:
2017-12-12 04:19:45 +00:00
return False
2016-12-20 17:03:52 +00:00
def to_json(self):
2018-03-23 11:46:46 +00:00
info = {
'id': self.id,
'hostname': self.hostname,
'ip': self.ip,
'port': self.port,
}
2018-03-23 11:46:46 +00:00
if self.domain and self.domain.gateway_set.all():
info["gateways"] = [d.id for d in self.domain.gateway_set.all()]
return info
2016-12-20 16:43:52 +00:00
def get_auth_info(self):
if self.admin_user:
return {
'username': self.admin_user.username,
'password': self.admin_user.password,
'private_key': self.admin_user.private_key_file,
'become': self.admin_user.become_info,
}
2018-10-30 04:06:39 +00:00
def as_node(self):
from .node import Node
fake_node = Node()
fake_node.id = self.id
fake_node.key = self.id
fake_node.value = self.hostname
fake_node.asset = self
fake_node.is_node = False
return fake_node
2017-03-05 03:38:02 +00:00
def _to_secret_json(self):
2017-12-07 08:25:50 +00:00
"""
2017-12-12 04:19:45 +00:00
Ansible use it create inventory, First using asset user,
otherwise using cluster admin user
2017-03-05 03:38:02 +00:00
2017-12-07 08:25:50 +00:00
Todo: May be move to ops implements it
"""
data = self.to_json()
if self.admin_user:
admin_user = self.admin_user
2017-12-07 05:01:33 +00:00
data.update({
2017-12-12 04:19:45 +00:00
'username': admin_user.username,
'password': admin_user.password,
'private_key': admin_user.private_key_file,
2017-12-28 06:25:56 +00:00
'become': admin_user.become_info,
'groups': [node.value for node in self.nodes.all()],
2017-12-07 05:01:33 +00:00
})
return data
2016-12-20 16:43:52 +00:00
class Meta:
2018-07-20 02:54:16 +00:00
unique_together = [('org_id', 'hostname')]
2018-01-05 09:57:02 +00:00
verbose_name = _("Asset")
2016-12-20 16:43:52 +00:00
@classmethod
def generate_fake(cls, count=100):
from random import seed, choice
import forgery_py
from django.db import IntegrityError
seed()
for i in range(count):
2018-05-28 07:00:06 +00:00
ip = [str(i) for i in random.sample(range(255), 4)]
asset = cls(ip='.'.join(ip),
2016-12-20 16:43:52 +00:00
hostname=forgery_py.internet.user_name(True),
admin_user=choice(AdminUser.objects.all()),
port=22,
created_by='Fake')
try:
asset.save()
asset.system_users = [choice(SystemUser.objects.all()) for i in range(3)]
logger.debug('Generate fake asset : %s' % asset.ip)
except IntegrityError:
print('Error continue')
continue