2017-05-24 11:33:11 -04:00
|
|
|
from __future__ import unicode_literals
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2017-04-03 14:00:15 -04:00
|
|
|
from django.core.exceptions import ObjectDoesNotExist
|
2017-05-24 11:33:11 -04:00
|
|
|
from rest_framework import serializers
|
2018-05-22 12:46:14 -04:00
|
|
|
from taggit.models import Tag
|
2017-05-24 11:33:11 -04:00
|
|
|
|
2017-04-03 14:00:15 -04:00
|
|
|
from dcim.api.serializers import NestedDeviceSerializer, NestedRackSerializer, NestedSiteSerializer
|
|
|
|
from dcim.models import Device, Rack, Site
|
2018-06-19 14:57:03 -04:00
|
|
|
from extras.models import ExportTemplate, Graph, ImageAttachment, ObjectChange, ReportResult, TopologyMap, UserAction
|
2018-05-30 11:19:10 -04:00
|
|
|
from extras.constants import *
|
2018-06-19 14:57:03 -04:00
|
|
|
from users.api.serializers import NestedUserSerializer
|
|
|
|
from utilities.api import (
|
|
|
|
ChoiceFieldSerializer, ContentTypeFieldSerializer, get_serializer_for_model, ValidatedModelSerializer,
|
|
|
|
)
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
|
2017-03-08 16:12:14 -05:00
|
|
|
#
|
|
|
|
# Graphs
|
|
|
|
#
|
|
|
|
|
2018-04-04 17:01:24 -04:00
|
|
|
class GraphSerializer(ValidatedModelSerializer):
|
2017-03-20 15:14:33 -04:00
|
|
|
type = ChoiceFieldSerializer(choices=GRAPH_TYPE_CHOICES)
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = Graph
|
|
|
|
fields = ['id', 'type', 'weight', 'name', 'source', 'link']
|
|
|
|
|
|
|
|
|
|
|
|
class RenderedGraphSerializer(serializers.ModelSerializer):
|
2016-03-01 11:23:03 -05:00
|
|
|
embed_url = serializers.SerializerMethodField()
|
2016-06-24 11:16:49 -04:00
|
|
|
embed_link = serializers.SerializerMethodField()
|
2017-03-20 15:14:33 -04:00
|
|
|
type = ChoiceFieldSerializer(choices=GRAPH_TYPE_CHOICES)
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = Graph
|
2017-03-20 15:14:33 -04:00
|
|
|
fields = ['id', 'type', 'weight', 'name', 'embed_url', 'embed_link']
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
def get_embed_url(self, obj):
|
|
|
|
return obj.embed_url(self.context['graphed_object'])
|
2016-06-24 11:16:49 -04:00
|
|
|
|
|
|
|
def get_embed_link(self, obj):
|
|
|
|
return obj.embed_link(self.context['graphed_object'])
|
2017-03-08 16:12:14 -05:00
|
|
|
|
|
|
|
|
2017-03-20 16:21:10 -04:00
|
|
|
#
|
|
|
|
# Export templates
|
|
|
|
#
|
|
|
|
|
2018-04-04 17:01:24 -04:00
|
|
|
class ExportTemplateSerializer(ValidatedModelSerializer):
|
2017-03-20 16:21:10 -04:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = ExportTemplate
|
|
|
|
fields = ['id', 'content_type', 'name', 'description', 'template_code', 'mime_type', 'file_extension']
|
|
|
|
|
|
|
|
|
2017-03-08 16:12:14 -05:00
|
|
|
#
|
|
|
|
# Topology maps
|
|
|
|
#
|
|
|
|
|
2018-04-04 17:01:24 -04:00
|
|
|
class TopologyMapSerializer(ValidatedModelSerializer):
|
2017-03-08 16:18:41 -05:00
|
|
|
site = NestedSiteSerializer()
|
2017-03-08 16:12:14 -05:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = TopologyMap
|
|
|
|
fields = ['id', 'name', 'slug', 'site', 'device_patterns', 'description']
|
|
|
|
|
|
|
|
|
2018-05-22 12:46:14 -04:00
|
|
|
#
|
|
|
|
# Tags
|
|
|
|
#
|
|
|
|
|
|
|
|
class TagSerializer(ValidatedModelSerializer):
|
|
|
|
tagged_items = serializers.IntegerField(read_only=True)
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = Tag
|
|
|
|
fields = ['id', 'name', 'slug', 'tagged_items']
|
|
|
|
|
|
|
|
|
2017-04-03 14:00:15 -04:00
|
|
|
#
|
|
|
|
# Image attachments
|
|
|
|
#
|
|
|
|
|
2018-04-04 17:01:24 -04:00
|
|
|
class ImageAttachmentSerializer(ValidatedModelSerializer):
|
2017-04-03 14:00:15 -04:00
|
|
|
content_type = ContentTypeFieldSerializer()
|
2018-04-04 17:01:24 -04:00
|
|
|
parent = serializers.SerializerMethodField(read_only=True)
|
2017-04-03 14:00:15 -04:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = ImageAttachment
|
2018-04-04 17:01:24 -04:00
|
|
|
fields = [
|
|
|
|
'id', 'content_type', 'object_id', 'parent', 'name', 'image', 'image_height', 'image_width', 'created',
|
|
|
|
]
|
2017-04-03 14:00:15 -04:00
|
|
|
|
|
|
|
def validate(self, data):
|
|
|
|
|
|
|
|
# Validate that the parent object exists
|
|
|
|
try:
|
|
|
|
data['content_type'].get_object_for_this_type(id=data['object_id'])
|
|
|
|
except ObjectDoesNotExist:
|
|
|
|
raise serializers.ValidationError(
|
|
|
|
"Invalid parent object: {} ID {}".format(data['content_type'], data['object_id'])
|
|
|
|
)
|
|
|
|
|
2017-07-06 17:37:24 -04:00
|
|
|
# Enforce model validation
|
2018-04-04 17:01:24 -04:00
|
|
|
super(ImageAttachmentSerializer, self).validate(data)
|
2017-07-06 17:37:24 -04:00
|
|
|
|
2017-04-03 14:00:15 -04:00
|
|
|
return data
|
|
|
|
|
2018-04-04 17:01:24 -04:00
|
|
|
def get_parent(self, obj):
|
|
|
|
|
|
|
|
# Static mapping of models to their nested serializers
|
|
|
|
if isinstance(obj.parent, Device):
|
|
|
|
serializer = NestedDeviceSerializer
|
|
|
|
elif isinstance(obj.parent, Rack):
|
|
|
|
serializer = NestedRackSerializer
|
|
|
|
elif isinstance(obj.parent, Site):
|
|
|
|
serializer = NestedSiteSerializer
|
|
|
|
else:
|
|
|
|
raise Exception("Unexpected type of parent object for ImageAttachment")
|
|
|
|
|
|
|
|
return serializer(obj.parent, context={'request': self.context['request']}).data
|
|
|
|
|
2017-04-03 14:00:15 -04:00
|
|
|
|
2017-09-25 17:27:58 -04:00
|
|
|
#
|
|
|
|
# Reports
|
|
|
|
#
|
|
|
|
|
|
|
|
class ReportResultSerializer(serializers.ModelSerializer):
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = ReportResult
|
|
|
|
fields = ['created', 'user', 'failed', 'data']
|
|
|
|
|
|
|
|
|
|
|
|
class NestedReportResultSerializer(serializers.ModelSerializer):
|
2017-09-26 16:55:25 -04:00
|
|
|
url = serializers.HyperlinkedIdentityField(
|
|
|
|
view_name='extras-api:report-detail',
|
|
|
|
lookup_field='report',
|
|
|
|
lookup_url_kwarg='pk'
|
|
|
|
)
|
2017-09-25 17:27:58 -04:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = ReportResult
|
2017-09-26 16:55:25 -04:00
|
|
|
fields = ['url', 'created', 'user', 'failed']
|
2017-09-25 17:27:58 -04:00
|
|
|
|
|
|
|
|
|
|
|
class ReportSerializer(serializers.Serializer):
|
|
|
|
module = serializers.CharField(max_length=255)
|
|
|
|
name = serializers.CharField(max_length=255)
|
|
|
|
description = serializers.CharField(max_length=255, required=False)
|
|
|
|
test_methods = serializers.ListField(child=serializers.CharField(max_length=255))
|
|
|
|
result = NestedReportResultSerializer()
|
|
|
|
|
|
|
|
|
|
|
|
class ReportDetailSerializer(ReportSerializer):
|
|
|
|
result = ReportResultSerializer()
|
|
|
|
|
|
|
|
|
2018-06-19 14:57:03 -04:00
|
|
|
#
|
|
|
|
# Change logging
|
|
|
|
#
|
|
|
|
|
|
|
|
class ObjectChangeSerializer(serializers.ModelSerializer):
|
|
|
|
user = NestedUserSerializer(read_only=True)
|
|
|
|
content_type = ContentTypeFieldSerializer(read_only=True)
|
|
|
|
changed_object = serializers.SerializerMethodField(read_only=True)
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = ObjectChange
|
2018-06-19 15:45:15 -04:00
|
|
|
fields = [
|
|
|
|
'id', 'time', 'user', 'user_name', 'request_id', 'action', 'content_type', 'changed_object', 'object_data',
|
|
|
|
]
|
2018-06-19 14:57:03 -04:00
|
|
|
|
|
|
|
def get_changed_object(self, obj):
|
|
|
|
"""
|
|
|
|
Serialize a nested representation of the changed object.
|
|
|
|
"""
|
2018-06-19 15:45:15 -04:00
|
|
|
if obj.changed_object is None:
|
|
|
|
return None
|
2018-06-19 14:57:03 -04:00
|
|
|
serializer = get_serializer_for_model(obj.changed_object, prefix='Nested')
|
|
|
|
if serializer is None:
|
|
|
|
return obj.object_repr
|
|
|
|
context = {'request': self.context['request']}
|
|
|
|
data = serializer(obj.changed_object, context=context).data
|
|
|
|
return data
|
|
|
|
|
|
|
|
|
2017-03-09 14:26:39 -05:00
|
|
|
#
|
|
|
|
# User actions
|
|
|
|
#
|
|
|
|
|
|
|
|
class UserActionSerializer(serializers.ModelSerializer):
|
|
|
|
user = NestedUserSerializer()
|
|
|
|
action = ChoiceFieldSerializer(choices=ACTION_CHOICES)
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = UserAction
|
|
|
|
fields = ['id', 'time', 'user', 'action', 'message']
|