mirror of https://github.com/jumpserver/jumpserver
194 lines
6.7 KiB
Python
194 lines
6.7 KiB
Python
# -*- coding: utf-8 -*-
|
|
#
|
|
from rest_framework import serializers
|
|
from rest_framework.validators import ValidationError
|
|
|
|
from django.utils.translation import ugettext_lazy as _
|
|
|
|
from orgs.mixins import BulkOrgResourceModelSerializer
|
|
from common.serializers import AdaptedBulkListSerializer
|
|
from ..models import Asset, Protocol
|
|
from .system_user import AssetSystemUserSerializer
|
|
|
|
__all__ = [
|
|
'AssetSerializer', 'AssetGrantedSerializer', 'AssetSimpleSerializer',
|
|
'ProtocolSerializer',
|
|
]
|
|
|
|
|
|
class ProtocolSerializer(serializers.ModelSerializer):
|
|
class Meta:
|
|
model = Protocol
|
|
fields = ["name", "port"]
|
|
|
|
|
|
class ProtocolsRelatedField(serializers.RelatedField):
|
|
def to_representation(self, value):
|
|
return str(value)
|
|
|
|
def to_internal_value(self, data):
|
|
if isinstance(data, dict):
|
|
return data
|
|
if '/' not in data:
|
|
raise ValidationError("protocol not contain /: {}".format(data))
|
|
v = data.split("/")
|
|
if len(v) != 2:
|
|
raise ValidationError("protocol format should be name/port: {}".format(data))
|
|
name, port = v
|
|
cleaned_data = {"name": name, "port": port}
|
|
return cleaned_data
|
|
|
|
|
|
class AssetSerializer(BulkOrgResourceModelSerializer):
|
|
protocols = ProtocolsRelatedField(
|
|
many=True, queryset=Protocol.objects.all(), label=_("Protocols")
|
|
)
|
|
|
|
"""
|
|
资产的数据结构
|
|
"""
|
|
class Meta:
|
|
model = Asset
|
|
list_serializer_class = AdaptedBulkListSerializer
|
|
fields = [
|
|
'id', 'ip', 'hostname', 'protocol', 'port',
|
|
'protocols', 'platform', 'is_active', 'public_ip', 'domain',
|
|
'admin_user', 'nodes', 'labels', 'number', 'vendor', 'model', 'sn',
|
|
'cpu_model', 'cpu_count', 'cpu_cores', 'cpu_vcpus', 'memory',
|
|
'disk_total', 'disk_info', 'os', 'os_version', 'os_arch',
|
|
'hostname_raw', 'comment', 'created_by', 'date_created',
|
|
'hardware_info', 'connectivity'
|
|
]
|
|
read_only_fields = (
|
|
'vendor', 'model', 'sn', 'cpu_model', 'cpu_count',
|
|
'cpu_cores', 'cpu_vcpus', 'memory', 'disk_total', 'disk_info',
|
|
'os', 'os_version', 'os_arch', 'hostname_raw',
|
|
'created_by', 'date_created',
|
|
)
|
|
extra_kwargs = {
|
|
'protocol': {'write_only': True},
|
|
'port': {'write_only': True},
|
|
'hardware_info': {'label': _('Hardware info')},
|
|
'connectivity': {'label': _('Connectivity')},
|
|
'org_name': {'label': _('Org name')}
|
|
}
|
|
|
|
@classmethod
|
|
def setup_eager_loading(cls, queryset):
|
|
""" Perform necessary eager loading of data. """
|
|
queryset = queryset.prefetch_related('labels', 'nodes')\
|
|
.select_related('admin_user')
|
|
return queryset
|
|
|
|
@staticmethod
|
|
def validate_protocols(attr):
|
|
protocols_serializer = ProtocolSerializer(data=attr, many=True)
|
|
protocols_serializer.is_valid(raise_exception=True)
|
|
protocols_name = [i.get("name", "ssh") for i in attr]
|
|
errors = [{} for i in protocols_name]
|
|
for i, name in enumerate(protocols_name):
|
|
if name in protocols_name[:i]:
|
|
errors[i] = {"name": _("Protocol duplicate: {}").format(name)}
|
|
if any(errors):
|
|
raise ValidationError(errors)
|
|
return attr
|
|
|
|
def create(self, validated_data):
|
|
protocols_data = validated_data.pop("protocols", [])
|
|
|
|
# 兼容老的api
|
|
protocol = validated_data.get("protocol")
|
|
port = validated_data.get("port")
|
|
if not protocols_data and protocol and port:
|
|
protocols_data = [{"name": protocol, "port": port}]
|
|
|
|
if not protocol and not port and protocols_data:
|
|
validated_data["protocol"] = protocols_data[0]["name"]
|
|
validated_data["port"] = protocols_data[0]["port"]
|
|
|
|
protocols_serializer = ProtocolSerializer(data=protocols_data, many=True)
|
|
protocols_serializer.is_valid(raise_exception=True)
|
|
protocols = protocols_serializer.save()
|
|
instance = super().create(validated_data)
|
|
instance.protocols.set(protocols)
|
|
return instance
|
|
|
|
def update(self, instance, validated_data):
|
|
protocols_data = validated_data.pop("protocols", [])
|
|
|
|
# 兼容老的api
|
|
protocol = validated_data.get("protocol")
|
|
port = validated_data.get("port")
|
|
if not protocols_data and protocol and port:
|
|
protocols_data = [{"name": protocol, "port": port}]
|
|
|
|
if not protocol and not port and protocols_data:
|
|
validated_data["protocol"] = protocols_data[0]["name"]
|
|
validated_data["port"] = protocols_data[0]["port"]
|
|
protocols = None
|
|
if protocols_data:
|
|
protocols_serializer = ProtocolSerializer(data=protocols_data, many=True)
|
|
protocols_serializer.is_valid(raise_exception=True)
|
|
protocols = protocols_serializer.save()
|
|
|
|
instance = super().update(instance, validated_data)
|
|
if protocols:
|
|
instance.protocols.all().delete()
|
|
instance.protocols.set(protocols)
|
|
return instance
|
|
|
|
|
|
# class AssetAsNodeSerializer(serializers.ModelSerializer):
|
|
# protocols = ProtocolSerializer(many=True)
|
|
#
|
|
# class Meta:
|
|
# model = Asset
|
|
# fields = ['id', 'hostname', 'ip', 'platform', 'protocols']
|
|
|
|
|
|
class AssetGrantedSerializer(serializers.ModelSerializer):
|
|
"""
|
|
被授权资产的数据结构
|
|
"""
|
|
protocols = ProtocolsRelatedField(
|
|
many=True, queryset=Protocol.objects.all(), label=_("Protocols")
|
|
)
|
|
system_users_granted = AssetSystemUserSerializer(many=True, read_only=True)
|
|
system_users_join = serializers.SerializerMethodField()
|
|
# nodes = NodeTMPSerializer(many=True, read_only=True)
|
|
|
|
class Meta:
|
|
model = Asset
|
|
fields = (
|
|
"id", "hostname", "ip", "protocol", "port", "protocols",
|
|
"system_users_granted", "is_active", "system_users_join", "os",
|
|
'domain', "platform", "comment", "org_id", "org_name",
|
|
)
|
|
|
|
@staticmethod
|
|
def get_system_users_join(obj):
|
|
system_users = [s.username for s in obj.system_users_granted]
|
|
return ', '.join(system_users)
|
|
|
|
|
|
# class MyAssetGrantedSerializer(AssetGrantedSerializer):
|
|
# """
|
|
# 普通用户获取授权的资产定义的数据结构
|
|
# """
|
|
# protocols = ProtocolSerializer(many=True)
|
|
#
|
|
# class Meta:
|
|
# model = Asset
|
|
# fields = (
|
|
# "id", "hostname", "system_users_granted",
|
|
# "is_active", "system_users_join", "org_name",
|
|
# "os", "platform", "comment", "org_id", "protocols"
|
|
# )
|
|
|
|
|
|
class AssetSimpleSerializer(serializers.ModelSerializer):
|
|
|
|
class Meta:
|
|
model = Asset
|
|
fields = ['id', 'hostname', 'ip', 'connectivity', 'port']
|