serializers.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290
  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.choices 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. ipaddress_count = serializers.IntegerField(read_only=True)
  24. prefix_count = serializers.IntegerField(read_only=True)
  25. class Meta:
  26. model = VRF
  27. fields = [
  28. 'id', 'name', 'rd', 'tenant', 'enforce_unique', 'description', 'tags', 'display_name', 'custom_fields',
  29. 'created', 'last_updated', 'ipaddress_count', 'prefix_count',
  30. ]
  31. #
  32. # RIRs/aggregates
  33. #
  34. class RIRSerializer(ValidatedModelSerializer):
  35. aggregate_count = serializers.IntegerField(read_only=True)
  36. class Meta:
  37. model = RIR
  38. fields = ['id', 'name', 'slug', 'is_private', 'description', 'aggregate_count']
  39. class AggregateSerializer(TaggitSerializer, CustomFieldModelSerializer):
  40. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  41. rir = NestedRIRSerializer()
  42. tags = TagListSerializerField(required=False)
  43. class Meta:
  44. model = Aggregate
  45. fields = [
  46. 'id', 'family', 'prefix', 'rir', 'date_added', 'description', 'tags', 'custom_fields', 'created',
  47. 'last_updated',
  48. ]
  49. read_only_fields = ['family']
  50. #
  51. # VLANs
  52. #
  53. class RoleSerializer(ValidatedModelSerializer):
  54. prefix_count = serializers.IntegerField(read_only=True)
  55. vlan_count = serializers.IntegerField(read_only=True)
  56. class Meta:
  57. model = Role
  58. fields = ['id', 'name', 'slug', 'weight', 'description', 'prefix_count', 'vlan_count']
  59. class VLANGroupSerializer(ValidatedModelSerializer):
  60. site = NestedSiteSerializer(required=False, allow_null=True)
  61. vlan_count = serializers.IntegerField(read_only=True)
  62. class Meta:
  63. model = VLANGroup
  64. fields = ['id', 'name', 'slug', 'site', 'description', 'vlan_count']
  65. validators = []
  66. def validate(self, data):
  67. # Validate uniqueness of name and slug if a site has been assigned.
  68. if data.get('site', None):
  69. for field in ['name', 'slug']:
  70. validator = UniqueTogetherValidator(queryset=VLANGroup.objects.all(), fields=('site', field))
  71. validator(data, self)
  72. # Enforce model validation
  73. super().validate(data)
  74. return data
  75. class VLANSerializer(TaggitSerializer, CustomFieldModelSerializer):
  76. site = NestedSiteSerializer(required=False, allow_null=True)
  77. group = NestedVLANGroupSerializer(required=False, allow_null=True)
  78. tenant = NestedTenantSerializer(required=False, allow_null=True)
  79. status = ChoiceField(choices=VLANStatusChoices, required=False)
  80. role = NestedRoleSerializer(required=False, allow_null=True)
  81. tags = TagListSerializerField(required=False)
  82. prefix_count = serializers.IntegerField(read_only=True)
  83. class Meta:
  84. model = VLAN
  85. fields = [
  86. 'id', 'site', 'group', 'vid', 'name', 'tenant', 'status', 'role', 'description', 'tags', 'display_name',
  87. 'custom_fields', 'created', 'last_updated', 'prefix_count',
  88. ]
  89. validators = []
  90. def validate(self, data):
  91. # Validate uniqueness of vid and name if a group has been assigned.
  92. if data.get('group', None):
  93. for field in ['vid', 'name']:
  94. validator = UniqueTogetherValidator(queryset=VLAN.objects.all(), fields=('group', field))
  95. validator(data, self)
  96. # Enforce model validation
  97. super().validate(data)
  98. return data
  99. #
  100. # Prefixes
  101. #
  102. class PrefixSerializer(TaggitSerializer, CustomFieldModelSerializer):
  103. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  104. site = NestedSiteSerializer(required=False, allow_null=True)
  105. vrf = NestedVRFSerializer(required=False, allow_null=True)
  106. tenant = NestedTenantSerializer(required=False, allow_null=True)
  107. vlan = NestedVLANSerializer(required=False, allow_null=True)
  108. status = ChoiceField(choices=PrefixStatusChoices, required=False)
  109. role = NestedRoleSerializer(required=False, allow_null=True)
  110. tags = TagListSerializerField(required=False)
  111. class Meta:
  112. model = Prefix
  113. fields = [
  114. 'id', 'family', 'prefix', 'site', 'vrf', 'tenant', 'vlan', 'status', 'role', 'is_pool', 'description',
  115. 'tags', 'custom_fields', 'created', 'last_updated',
  116. ]
  117. read_only_fields = ['family']
  118. class PrefixLengthSerializer(serializers.Serializer):
  119. prefix_length = serializers.IntegerField()
  120. def to_internal_value(self, data):
  121. requested_prefix = data.get('prefix_length')
  122. if requested_prefix is None:
  123. raise serializers.ValidationError({
  124. 'prefix_length': 'this field can not be missing'
  125. })
  126. if not isinstance(requested_prefix, int):
  127. raise serializers.ValidationError({
  128. 'prefix_length': 'this field must be int type'
  129. })
  130. prefix = self.context.get('prefix')
  131. if prefix.family == 4 and requested_prefix > 32:
  132. raise serializers.ValidationError({
  133. 'prefix_length': 'Invalid prefix length ({}) for IPv4'.format((requested_prefix))
  134. })
  135. elif prefix.family == 6 and requested_prefix > 128:
  136. raise serializers.ValidationError({
  137. 'prefix_length': 'Invalid prefix length ({}) for IPv6'.format((requested_prefix))
  138. })
  139. return data
  140. class AvailablePrefixSerializer(serializers.Serializer):
  141. """
  142. Representation of a prefix which does not exist in the database.
  143. """
  144. family = serializers.IntegerField(read_only=True)
  145. prefix = serializers.CharField(read_only=True)
  146. vrf = NestedVRFSerializer(read_only=True)
  147. def to_representation(self, instance):
  148. if self.context.get('vrf'):
  149. vrf = NestedVRFSerializer(self.context['vrf'], context={'request': self.context['request']}).data
  150. else:
  151. vrf = None
  152. return OrderedDict([
  153. ('family', instance.version),
  154. ('prefix', str(instance)),
  155. ('vrf', vrf),
  156. ])
  157. #
  158. # IP addresses
  159. #
  160. class IPAddressInterfaceSerializer(WritableNestedSerializer):
  161. """
  162. Nested representation of an Interface which may belong to a Device *or* a VirtualMachine.
  163. """
  164. url = serializers.SerializerMethodField() # We're imitating a HyperlinkedIdentityField here
  165. device = NestedDeviceSerializer(read_only=True)
  166. virtual_machine = NestedVirtualMachineSerializer(read_only=True)
  167. class Meta:
  168. model = Interface
  169. fields = [
  170. 'id', 'url', 'device', 'virtual_machine', 'name',
  171. ]
  172. def get_url(self, obj):
  173. """
  174. Return a link to the Interface via either the DCIM API if the parent is a Device, or via the virtualization API
  175. if the parent is a VirtualMachine.
  176. """
  177. url_name = 'dcim-api:interface-detail' if obj.device else 'virtualization-api:interface-detail'
  178. return reverse(url_name, kwargs={'pk': obj.pk}, request=self.context['request'])
  179. class IPAddressSerializer(TaggitSerializer, CustomFieldModelSerializer):
  180. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  181. vrf = NestedVRFSerializer(required=False, allow_null=True)
  182. tenant = NestedTenantSerializer(required=False, allow_null=True)
  183. status = ChoiceField(choices=IPAddressStatusChoices, required=False)
  184. role = ChoiceField(choices=IPAddressRoleChoices, allow_blank=True, required=False)
  185. interface = IPAddressInterfaceSerializer(required=False, allow_null=True)
  186. nat_inside = NestedIPAddressSerializer(required=False, allow_null=True)
  187. nat_outside = NestedIPAddressSerializer(read_only=True)
  188. tags = TagListSerializerField(required=False)
  189. class Meta:
  190. model = IPAddress
  191. fields = [
  192. 'id', 'family', 'address', 'vrf', 'tenant', 'status', 'role', 'interface', 'nat_inside',
  193. 'nat_outside', 'dns_name', 'description', 'tags', 'custom_fields', 'created', 'last_updated',
  194. ]
  195. read_only_fields = ['family']
  196. class AvailableIPSerializer(serializers.Serializer):
  197. """
  198. Representation of an IP address which does not exist in the database.
  199. """
  200. family = serializers.IntegerField(read_only=True)
  201. address = serializers.CharField(read_only=True)
  202. vrf = NestedVRFSerializer(read_only=True)
  203. def to_representation(self, instance):
  204. if self.context.get('vrf'):
  205. vrf = NestedVRFSerializer(self.context['vrf'], context={'request': self.context['request']}).data
  206. else:
  207. vrf = None
  208. return OrderedDict([
  209. ('family', self.context['prefix'].version),
  210. ('address', '{}/{}'.format(instance, self.context['prefix'].prefixlen)),
  211. ('vrf', vrf),
  212. ])
  213. #
  214. # Services
  215. #
  216. class ServiceSerializer(TaggitSerializer, CustomFieldModelSerializer):
  217. device = NestedDeviceSerializer(required=False, allow_null=True)
  218. virtual_machine = NestedVirtualMachineSerializer(required=False, allow_null=True)
  219. protocol = ChoiceField(choices=ServiceProtocolChoices, required=False)
  220. ipaddresses = SerializedPKRelatedField(
  221. queryset=IPAddress.objects.all(),
  222. serializer=NestedIPAddressSerializer,
  223. required=False,
  224. many=True
  225. )
  226. tags = TagListSerializerField(required=False)
  227. class Meta:
  228. model = Service
  229. fields = [
  230. 'id', 'device', 'virtual_machine', 'name', 'port', 'protocol', 'ipaddresses', 'description', 'tags',
  231. 'custom_fields', 'created', 'last_updated',
  232. ]