serializers.py 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. from collections import OrderedDict
  2. from rest_framework import serializers
  3. from rest_framework.reverse import reverse
  4. from rest_framework.validators import UniqueTogetherValidator
  5. from taggit_serializer.serializers import TaggitSerializer, TagListSerializerField
  6. from dcim.api.nested_serializers import NestedDeviceSerializer, NestedSiteSerializer
  7. from dcim.models import Interface
  8. from extras.api.customfields import CustomFieldModelSerializer
  9. from ipam.constants import *
  10. from ipam.models import Aggregate, IPAddress, Prefix, RIR, Role, Service, VLAN, VLANGroup, VRF
  11. from tenancy.api.nested_serializers import NestedTenantSerializer
  12. from utilities.api import (
  13. ChoiceField, SerializedPKRelatedField, ValidatedModelSerializer, WritableNestedSerializer,
  14. )
  15. from virtualization.api.nested_serializers import NestedVirtualMachineSerializer
  16. from .nested_serializers import *
  17. #
  18. # VRFs
  19. #
  20. class VRFSerializer(TaggitSerializer, CustomFieldModelSerializer):
  21. tenant = NestedTenantSerializer(required=False, allow_null=True)
  22. tags = TagListSerializerField(required=False)
  23. class Meta:
  24. model = VRF
  25. fields = [
  26. 'id', 'name', 'rd', 'tenant', 'enforce_unique', 'description', 'tags', 'display_name', 'custom_fields',
  27. 'created', 'last_updated',
  28. ]
  29. #
  30. # RIRs/aggregates
  31. #
  32. class RIRSerializer(ValidatedModelSerializer):
  33. class Meta:
  34. model = RIR
  35. fields = ['id', 'name', 'slug', 'is_private']
  36. class AggregateSerializer(TaggitSerializer, CustomFieldModelSerializer):
  37. rir = NestedRIRSerializer()
  38. tags = TagListSerializerField(required=False)
  39. class Meta:
  40. model = Aggregate
  41. fields = [
  42. 'id', 'family', 'prefix', 'rir', 'date_added', 'description', 'tags', 'custom_fields', 'created',
  43. 'last_updated',
  44. ]
  45. read_only_fields = ['family']
  46. #
  47. # VLANs
  48. #
  49. class RoleSerializer(ValidatedModelSerializer):
  50. class Meta:
  51. model = Role
  52. fields = ['id', 'name', 'slug', 'weight']
  53. class VLANGroupSerializer(ValidatedModelSerializer):
  54. site = NestedSiteSerializer(required=False, allow_null=True)
  55. class Meta:
  56. model = VLANGroup
  57. fields = ['id', 'name', 'slug', 'site']
  58. validators = []
  59. def validate(self, data):
  60. # Validate uniqueness of name and slug if a site has been assigned.
  61. if data.get('site', None):
  62. for field in ['name', 'slug']:
  63. validator = UniqueTogetherValidator(queryset=VLANGroup.objects.all(), fields=('site', field))
  64. validator.set_context(self)
  65. validator(data)
  66. # Enforce model validation
  67. super().validate(data)
  68. return data
  69. class VLANSerializer(TaggitSerializer, CustomFieldModelSerializer):
  70. site = NestedSiteSerializer(required=False, allow_null=True)
  71. group = NestedVLANGroupSerializer(required=False, allow_null=True)
  72. tenant = NestedTenantSerializer(required=False, allow_null=True)
  73. status = ChoiceField(choices=VLAN_STATUS_CHOICES, required=False)
  74. role = NestedRoleSerializer(required=False, allow_null=True)
  75. tags = TagListSerializerField(required=False)
  76. class Meta:
  77. model = VLAN
  78. fields = [
  79. 'id', 'site', 'group', 'vid', 'name', 'tenant', 'status', 'role', 'description', 'tags', 'display_name',
  80. 'custom_fields', 'created', 'last_updated',
  81. ]
  82. validators = []
  83. def validate(self, data):
  84. # Validate uniqueness of vid and name if a group has been assigned.
  85. if data.get('group', None):
  86. for field in ['vid', 'name']:
  87. validator = UniqueTogetherValidator(queryset=VLAN.objects.all(), fields=('group', field))
  88. validator.set_context(self)
  89. validator(data)
  90. # Enforce model validation
  91. super().validate(data)
  92. return data
  93. #
  94. # Prefixes
  95. #
  96. class PrefixSerializer(TaggitSerializer, CustomFieldModelSerializer):
  97. family = ChoiceField(choices=AF_CHOICES, read_only=True)
  98. site = NestedSiteSerializer(required=False, allow_null=True)
  99. vrf = NestedVRFSerializer(required=False, allow_null=True)
  100. tenant = NestedTenantSerializer(required=False, allow_null=True)
  101. vlan = NestedVLANSerializer(required=False, allow_null=True)
  102. status = ChoiceField(choices=PREFIX_STATUS_CHOICES, required=False)
  103. role = NestedRoleSerializer(required=False, allow_null=True)
  104. tags = TagListSerializerField(required=False)
  105. class Meta:
  106. model = Prefix
  107. fields = [
  108. 'id', 'family', 'prefix', 'site', 'vrf', 'tenant', 'vlan', 'status', 'role', 'is_pool', 'description',
  109. 'tags', 'custom_fields', 'created', 'last_updated',
  110. ]
  111. read_only_fields = ['family']
  112. class AvailablePrefixSerializer(serializers.Serializer):
  113. """
  114. Representation of a prefix which does not exist in the database.
  115. """
  116. def to_representation(self, instance):
  117. if self.context.get('vrf'):
  118. vrf = NestedVRFSerializer(self.context['vrf'], context={'request': self.context['request']}).data
  119. else:
  120. vrf = None
  121. return OrderedDict([
  122. ('family', instance.version),
  123. ('prefix', str(instance)),
  124. ('vrf', vrf),
  125. ])
  126. #
  127. # IP addresses
  128. #
  129. class IPAddressInterfaceSerializer(WritableNestedSerializer):
  130. """
  131. Nested representation of an Interface which may belong to a Device *or* a VirtualMachine.
  132. """
  133. url = serializers.SerializerMethodField() # We're imitating a HyperlinkedIdentityField here
  134. device = NestedDeviceSerializer(read_only=True)
  135. virtual_machine = NestedVirtualMachineSerializer(read_only=True)
  136. class Meta:
  137. model = Interface
  138. fields = [
  139. 'id', 'url', 'device', 'virtual_machine', 'name',
  140. ]
  141. def get_url(self, obj):
  142. """
  143. Return a link to the Interface via either the DCIM API if the parent is a Device, or via the virtualization API
  144. if the parent is a VirtualMachine.
  145. """
  146. url_name = 'dcim-api:interface-detail' if obj.device else 'virtualization-api:interface-detail'
  147. return reverse(url_name, kwargs={'pk': obj.pk}, request=self.context['request'])
  148. class IPAddressSerializer(TaggitSerializer, CustomFieldModelSerializer):
  149. family = ChoiceField(choices=AF_CHOICES, read_only=True)
  150. vrf = NestedVRFSerializer(required=False, allow_null=True)
  151. tenant = NestedTenantSerializer(required=False, allow_null=True)
  152. status = ChoiceField(choices=IPADDRESS_STATUS_CHOICES, required=False)
  153. role = ChoiceField(choices=IPADDRESS_ROLE_CHOICES, required=False, allow_null=True)
  154. interface = IPAddressInterfaceSerializer(required=False, allow_null=True)
  155. nat_inside = NestedIPAddressSerializer(required=False, allow_null=True)
  156. nat_outside = NestedIPAddressSerializer(read_only=True)
  157. tags = TagListSerializerField(required=False)
  158. class Meta:
  159. model = IPAddress
  160. fields = [
  161. 'id', 'family', 'address', 'vrf', 'tenant', 'status', 'role', 'interface', 'description', 'nat_inside',
  162. 'nat_outside', 'tags', 'custom_fields', 'created', 'last_updated',
  163. ]
  164. read_only_fields = ['family']
  165. class AvailableIPSerializer(serializers.Serializer):
  166. """
  167. Representation of an IP address which does not exist in the database.
  168. """
  169. def to_representation(self, instance):
  170. if self.context.get('vrf'):
  171. vrf = NestedVRFSerializer(self.context['vrf'], context={'request': self.context['request']}).data
  172. else:
  173. vrf = None
  174. return OrderedDict([
  175. ('family', self.context['prefix'].version),
  176. ('address', '{}/{}'.format(instance, self.context['prefix'].prefixlen)),
  177. ('vrf', vrf),
  178. ])
  179. #
  180. # Services
  181. #
  182. class ServiceSerializer(CustomFieldModelSerializer):
  183. device = NestedDeviceSerializer(required=False, allow_null=True)
  184. virtual_machine = NestedVirtualMachineSerializer(required=False, allow_null=True)
  185. protocol = ChoiceField(choices=IP_PROTOCOL_CHOICES)
  186. ipaddresses = SerializedPKRelatedField(
  187. queryset=IPAddress.objects.all(),
  188. serializer=NestedIPAddressSerializer,
  189. required=False,
  190. many=True
  191. )
  192. class Meta:
  193. model = Service
  194. fields = [
  195. 'id', 'device', 'virtual_machine', 'name', 'port', 'protocol', 'ipaddresses', 'description',
  196. 'custom_fields', 'created', 'last_updated',
  197. ]