serializers.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322
  1. from collections import OrderedDict
  2. from django.contrib.contenttypes.models import ContentType
  3. from drf_yasg.utils import swagger_serializer_method
  4. from rest_framework import serializers
  5. from rest_framework.validators import UniqueTogetherValidator
  6. from dcim.api.nested_serializers import NestedDeviceSerializer, NestedSiteSerializer
  7. from extras.api.customfields import CustomFieldModelSerializer
  8. from extras.api.serializers import TaggedObjectSerializer
  9. from ipam.choices import *
  10. from ipam.constants import IPADDRESS_ASSIGNMENT_MODELS
  11. from ipam.models import Aggregate, IPAddress, Prefix, RIR, Role, RouteTarget, Service, VLAN, VLANGroup, VRF
  12. from netbox.api import ChoiceField, ContentTypeField, SerializedPKRelatedField
  13. from netbox.api.serializers import OrganizationalModelSerializer
  14. from tenancy.api.nested_serializers import NestedTenantSerializer
  15. from utilities.api import get_serializer_for_model
  16. from virtualization.api.nested_serializers import NestedVirtualMachineSerializer
  17. from .nested_serializers import *
  18. #
  19. # VRFs
  20. #
  21. class VRFSerializer(TaggedObjectSerializer, CustomFieldModelSerializer):
  22. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:vrf-detail')
  23. tenant = NestedTenantSerializer(required=False, allow_null=True)
  24. import_targets = SerializedPKRelatedField(
  25. queryset=RouteTarget.objects.all(),
  26. serializer=NestedRouteTargetSerializer,
  27. required=False,
  28. many=True
  29. )
  30. export_targets = SerializedPKRelatedField(
  31. queryset=RouteTarget.objects.all(),
  32. serializer=NestedRouteTargetSerializer,
  33. required=False,
  34. many=True
  35. )
  36. ipaddress_count = serializers.IntegerField(read_only=True)
  37. prefix_count = serializers.IntegerField(read_only=True)
  38. class Meta:
  39. model = VRF
  40. fields = [
  41. 'id', 'url', 'name', 'rd', 'tenant', 'enforce_unique', 'description', 'import_targets', 'export_targets',
  42. 'tags', 'display_name', 'custom_fields', 'created', 'last_updated', 'ipaddress_count', 'prefix_count',
  43. ]
  44. #
  45. # Route targets
  46. #
  47. class RouteTargetSerializer(TaggedObjectSerializer, CustomFieldModelSerializer):
  48. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:routetarget-detail')
  49. tenant = NestedTenantSerializer(required=False, allow_null=True)
  50. class Meta:
  51. model = RouteTarget
  52. fields = [
  53. 'id', 'url', 'name', 'tenant', 'description', 'tags', 'custom_fields', 'created', 'last_updated',
  54. ]
  55. #
  56. # RIRs/aggregates
  57. #
  58. class RIRSerializer(OrganizationalModelSerializer):
  59. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:rir-detail')
  60. aggregate_count = serializers.IntegerField(read_only=True)
  61. class Meta:
  62. model = RIR
  63. fields = [
  64. 'id', 'url', 'name', 'slug', 'is_private', 'description', 'custom_fields', 'created', 'last_updated',
  65. 'aggregate_count',
  66. ]
  67. class AggregateSerializer(TaggedObjectSerializer, CustomFieldModelSerializer):
  68. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:aggregate-detail')
  69. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  70. rir = NestedRIRSerializer()
  71. tenant = NestedTenantSerializer(required=False, allow_null=True)
  72. class Meta:
  73. model = Aggregate
  74. fields = [
  75. 'id', 'url', 'family', 'prefix', 'rir', 'tenant', 'date_added', 'description', 'tags', 'custom_fields',
  76. 'created', 'last_updated',
  77. ]
  78. read_only_fields = ['family']
  79. #
  80. # VLANs
  81. #
  82. class RoleSerializer(OrganizationalModelSerializer):
  83. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:role-detail')
  84. prefix_count = serializers.IntegerField(read_only=True)
  85. vlan_count = serializers.IntegerField(read_only=True)
  86. class Meta:
  87. model = Role
  88. fields = [
  89. 'id', 'url', 'name', 'slug', 'weight', 'description', 'custom_fields', 'created', 'last_updated',
  90. 'prefix_count', 'vlan_count',
  91. ]
  92. class VLANGroupSerializer(OrganizationalModelSerializer):
  93. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:vlangroup-detail')
  94. site = NestedSiteSerializer(required=False, allow_null=True)
  95. vlan_count = serializers.IntegerField(read_only=True)
  96. class Meta:
  97. model = VLANGroup
  98. fields = [
  99. 'id', 'url', 'name', 'slug', 'site', 'description', 'custom_fields', 'created', 'last_updated',
  100. 'vlan_count',
  101. ]
  102. validators = []
  103. def validate(self, data):
  104. # Validate uniqueness of name and slug if a site has been assigned.
  105. if data.get('site', None):
  106. for field in ['name', 'slug']:
  107. validator = UniqueTogetherValidator(queryset=VLANGroup.objects.all(), fields=('site', field))
  108. validator(data, self)
  109. # Enforce model validation
  110. super().validate(data)
  111. return data
  112. class VLANSerializer(TaggedObjectSerializer, CustomFieldModelSerializer):
  113. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:vlan-detail')
  114. site = NestedSiteSerializer(required=False, allow_null=True)
  115. group = NestedVLANGroupSerializer(required=False, allow_null=True)
  116. tenant = NestedTenantSerializer(required=False, allow_null=True)
  117. status = ChoiceField(choices=VLANStatusChoices, required=False)
  118. role = NestedRoleSerializer(required=False, allow_null=True)
  119. prefix_count = serializers.IntegerField(read_only=True)
  120. class Meta:
  121. model = VLAN
  122. fields = [
  123. 'id', 'url', 'site', 'group', 'vid', 'name', 'tenant', 'status', 'role', 'description', 'tags',
  124. 'display_name', 'custom_fields', 'created', 'last_updated', 'prefix_count',
  125. ]
  126. validators = []
  127. def validate(self, data):
  128. # Validate uniqueness of vid and name if a group has been assigned.
  129. if data.get('group', None):
  130. for field in ['vid', 'name']:
  131. validator = UniqueTogetherValidator(queryset=VLAN.objects.all(), fields=('group', field))
  132. validator(data, self)
  133. # Enforce model validation
  134. super().validate(data)
  135. return data
  136. #
  137. # Prefixes
  138. #
  139. class PrefixSerializer(TaggedObjectSerializer, CustomFieldModelSerializer):
  140. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:prefix-detail')
  141. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  142. site = NestedSiteSerializer(required=False, allow_null=True)
  143. vrf = NestedVRFSerializer(required=False, allow_null=True)
  144. tenant = NestedTenantSerializer(required=False, allow_null=True)
  145. vlan = NestedVLANSerializer(required=False, allow_null=True)
  146. status = ChoiceField(choices=PrefixStatusChoices, required=False)
  147. role = NestedRoleSerializer(required=False, allow_null=True)
  148. class Meta:
  149. model = Prefix
  150. fields = [
  151. 'id', 'url', 'family', 'prefix', 'site', 'vrf', 'tenant', 'vlan', 'status', 'role', 'is_pool',
  152. 'description', 'tags', 'custom_fields', 'created', 'last_updated',
  153. ]
  154. read_only_fields = ['family']
  155. class PrefixLengthSerializer(serializers.Serializer):
  156. prefix_length = serializers.IntegerField()
  157. def to_internal_value(self, data):
  158. requested_prefix = data.get('prefix_length')
  159. if requested_prefix is None:
  160. raise serializers.ValidationError({
  161. 'prefix_length': 'this field can not be missing'
  162. })
  163. if not isinstance(requested_prefix, int):
  164. raise serializers.ValidationError({
  165. 'prefix_length': 'this field must be int type'
  166. })
  167. prefix = self.context.get('prefix')
  168. if prefix.family == 4 and requested_prefix > 32:
  169. raise serializers.ValidationError({
  170. 'prefix_length': 'Invalid prefix length ({}) for IPv4'.format((requested_prefix))
  171. })
  172. elif prefix.family == 6 and requested_prefix > 128:
  173. raise serializers.ValidationError({
  174. 'prefix_length': 'Invalid prefix length ({}) for IPv6'.format((requested_prefix))
  175. })
  176. return data
  177. class AvailablePrefixSerializer(serializers.Serializer):
  178. """
  179. Representation of a prefix which does not exist in the database.
  180. """
  181. family = serializers.IntegerField(read_only=True)
  182. prefix = serializers.CharField(read_only=True)
  183. vrf = NestedVRFSerializer(read_only=True)
  184. def to_representation(self, instance):
  185. if self.context.get('vrf'):
  186. vrf = NestedVRFSerializer(self.context['vrf'], context={'request': self.context['request']}).data
  187. else:
  188. vrf = None
  189. return OrderedDict([
  190. ('family', instance.version),
  191. ('prefix', str(instance)),
  192. ('vrf', vrf),
  193. ])
  194. #
  195. # IP addresses
  196. #
  197. class IPAddressSerializer(TaggedObjectSerializer, CustomFieldModelSerializer):
  198. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:ipaddress-detail')
  199. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  200. vrf = NestedVRFSerializer(required=False, allow_null=True)
  201. tenant = NestedTenantSerializer(required=False, allow_null=True)
  202. status = ChoiceField(choices=IPAddressStatusChoices, required=False)
  203. role = ChoiceField(choices=IPAddressRoleChoices, allow_blank=True, required=False)
  204. assigned_object_type = ContentTypeField(
  205. queryset=ContentType.objects.filter(IPADDRESS_ASSIGNMENT_MODELS),
  206. required=False,
  207. allow_null=True
  208. )
  209. assigned_object = serializers.SerializerMethodField(read_only=True)
  210. nat_inside = NestedIPAddressSerializer(required=False, allow_null=True)
  211. nat_outside = NestedIPAddressSerializer(read_only=True)
  212. class Meta:
  213. model = IPAddress
  214. fields = [
  215. 'id', 'url', 'family', 'address', 'vrf', 'tenant', 'status', 'role', 'assigned_object_type',
  216. 'assigned_object_id', 'assigned_object', 'nat_inside', 'nat_outside', 'dns_name', 'description', 'tags',
  217. 'custom_fields', 'created', 'last_updated',
  218. ]
  219. read_only_fields = ['family']
  220. @swagger_serializer_method(serializer_or_field=serializers.DictField)
  221. def get_assigned_object(self, obj):
  222. if obj.assigned_object is None:
  223. return None
  224. serializer = get_serializer_for_model(obj.assigned_object, prefix='Nested')
  225. context = {'request': self.context['request']}
  226. return serializer(obj.assigned_object, context=context).data
  227. class AvailableIPSerializer(serializers.Serializer):
  228. """
  229. Representation of an IP address which does not exist in the database.
  230. """
  231. family = serializers.IntegerField(read_only=True)
  232. address = serializers.CharField(read_only=True)
  233. vrf = NestedVRFSerializer(read_only=True)
  234. def to_representation(self, instance):
  235. if self.context.get('vrf'):
  236. vrf = NestedVRFSerializer(self.context['vrf'], context={'request': self.context['request']}).data
  237. else:
  238. vrf = None
  239. return OrderedDict([
  240. ('family', self.context['prefix'].version),
  241. ('address', '{}/{}'.format(instance, self.context['prefix'].prefixlen)),
  242. ('vrf', vrf),
  243. ])
  244. #
  245. # Services
  246. #
  247. class ServiceSerializer(TaggedObjectSerializer, CustomFieldModelSerializer):
  248. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:service-detail')
  249. device = NestedDeviceSerializer(required=False, allow_null=True)
  250. virtual_machine = NestedVirtualMachineSerializer(required=False, allow_null=True)
  251. protocol = ChoiceField(choices=ServiceProtocolChoices, required=False)
  252. ipaddresses = SerializedPKRelatedField(
  253. queryset=IPAddress.objects.all(),
  254. serializer=NestedIPAddressSerializer,
  255. required=False,
  256. many=True
  257. )
  258. class Meta:
  259. model = Service
  260. fields = [
  261. 'id', 'url', 'device', 'virtual_machine', 'name', 'ports', 'protocol', 'ipaddresses', 'description', 'tags',
  262. 'custom_fields', 'created', 'last_updated',
  263. ]