ip.py 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. from django.contrib.contenttypes.models import ContentType
  2. from drf_spectacular.utils import extend_schema_field
  3. from rest_framework import serializers
  4. from dcim.api.serializers_.sites import SiteSerializer
  5. from ipam.choices import *
  6. from ipam.constants import IPADDRESS_ASSIGNMENT_MODELS
  7. from ipam.models import Aggregate, IPAddress, IPRange, Prefix
  8. from netbox.api.fields import ChoiceField, ContentTypeField
  9. from netbox.api.serializers import NetBoxModelSerializer
  10. from tenancy.api.serializers_.tenants import TenantSerializer
  11. from utilities.api import get_serializer_for_model
  12. from .asns import RIRSerializer
  13. from .roles import RoleSerializer
  14. from .vlans import VLANSerializer
  15. from .vrfs import VRFSerializer
  16. from ..field_serializers import IPAddressField, IPNetworkField
  17. from ..nested_serializers import *
  18. __all__ = (
  19. 'AggregateSerializer',
  20. 'AvailableIPSerializer',
  21. 'AvailablePrefixSerializer',
  22. 'IPAddressSerializer',
  23. 'IPRangeSerializer',
  24. 'PrefixLengthSerializer',
  25. 'PrefixSerializer',
  26. )
  27. class AggregateSerializer(NetBoxModelSerializer):
  28. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:aggregate-detail')
  29. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  30. rir = RIRSerializer(nested=True)
  31. tenant = TenantSerializer(nested=True, required=False, allow_null=True)
  32. prefix = IPNetworkField()
  33. class Meta:
  34. model = Aggregate
  35. fields = [
  36. 'id', 'url', 'display', 'family', 'prefix', 'rir', 'tenant', 'date_added', 'description', 'comments',
  37. 'tags', 'custom_fields', 'created', 'last_updated',
  38. ]
  39. brief_fields = ('id', 'url', 'display', 'family', 'prefix', 'description')
  40. class PrefixSerializer(NetBoxModelSerializer):
  41. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:prefix-detail')
  42. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  43. site = SiteSerializer(nested=True, required=False, allow_null=True)
  44. vrf = VRFSerializer(nested=True, required=False, allow_null=True)
  45. tenant = TenantSerializer(nested=True, required=False, allow_null=True)
  46. vlan = VLANSerializer(nested=True, required=False, allow_null=True)
  47. status = ChoiceField(choices=PrefixStatusChoices, required=False)
  48. role = RoleSerializer(nested=True, required=False, allow_null=True)
  49. children = serializers.IntegerField(read_only=True)
  50. _depth = serializers.IntegerField(read_only=True)
  51. prefix = IPNetworkField()
  52. class Meta:
  53. model = Prefix
  54. fields = [
  55. 'id', 'url', 'display', 'family', 'prefix', 'site', 'vrf', 'tenant', 'vlan', 'status', 'role', 'is_pool',
  56. 'mark_utilized', 'description', 'comments', 'tags', 'custom_fields', 'created', 'last_updated', 'children',
  57. '_depth',
  58. ]
  59. brief_fields = ('id', 'url', 'display', 'family', 'prefix', 'description', '_depth')
  60. class PrefixLengthSerializer(serializers.Serializer):
  61. prefix_length = serializers.IntegerField()
  62. def to_internal_value(self, data):
  63. requested_prefix = data.get('prefix_length')
  64. if requested_prefix is None:
  65. raise serializers.ValidationError({
  66. 'prefix_length': 'this field can not be missing'
  67. })
  68. if not isinstance(requested_prefix, int):
  69. raise serializers.ValidationError({
  70. 'prefix_length': 'this field must be int type'
  71. })
  72. prefix = self.context.get('prefix')
  73. if prefix.family == 4 and requested_prefix > 32:
  74. raise serializers.ValidationError({
  75. 'prefix_length': 'Invalid prefix length ({}) for IPv4'.format(requested_prefix)
  76. })
  77. elif prefix.family == 6 and requested_prefix > 128:
  78. raise serializers.ValidationError({
  79. 'prefix_length': 'Invalid prefix length ({}) for IPv6'.format(requested_prefix)
  80. })
  81. return data
  82. class AvailablePrefixSerializer(serializers.Serializer):
  83. """
  84. Representation of a prefix which does not exist in the database.
  85. """
  86. family = serializers.IntegerField(read_only=True)
  87. prefix = serializers.CharField(read_only=True)
  88. vrf = VRFSerializer(nested=True, read_only=True, allow_null=True)
  89. def to_representation(self, instance):
  90. if self.context.get('vrf'):
  91. vrf = VRFSerializer(self.context['vrf'], nested=True, context={'request': self.context['request']}).data
  92. else:
  93. vrf = None
  94. return {
  95. 'family': instance.version,
  96. 'prefix': str(instance),
  97. 'vrf': vrf,
  98. }
  99. #
  100. # IP ranges
  101. #
  102. class IPRangeSerializer(NetBoxModelSerializer):
  103. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:iprange-detail')
  104. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  105. start_address = IPAddressField()
  106. end_address = IPAddressField()
  107. vrf = VRFSerializer(nested=True, required=False, allow_null=True)
  108. tenant = TenantSerializer(nested=True, required=False, allow_null=True)
  109. status = ChoiceField(choices=IPRangeStatusChoices, required=False)
  110. role = RoleSerializer(nested=True, required=False, allow_null=True)
  111. class Meta:
  112. model = IPRange
  113. fields = [
  114. 'id', 'url', 'display', 'family', 'start_address', 'end_address', 'size', 'vrf', 'tenant', 'status', 'role',
  115. 'description', 'comments', 'tags', 'custom_fields', 'created', 'last_updated',
  116. 'mark_utilized', 'description', 'comments', 'tags', 'custom_fields', 'created', 'last_updated',
  117. ]
  118. brief_fields = ('id', 'url', 'display', 'family', 'start_address', 'end_address', 'description')
  119. #
  120. # IP addresses
  121. #
  122. class IPAddressSerializer(NetBoxModelSerializer):
  123. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:ipaddress-detail')
  124. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  125. address = IPAddressField()
  126. vrf = VRFSerializer(nested=True, required=False, allow_null=True)
  127. tenant = TenantSerializer(nested=True, required=False, allow_null=True)
  128. status = ChoiceField(choices=IPAddressStatusChoices, required=False)
  129. role = ChoiceField(choices=IPAddressRoleChoices, allow_blank=True, required=False)
  130. assigned_object_type = ContentTypeField(
  131. queryset=ContentType.objects.filter(IPADDRESS_ASSIGNMENT_MODELS),
  132. required=False,
  133. allow_null=True
  134. )
  135. assigned_object = serializers.SerializerMethodField(read_only=True)
  136. nat_inside = NestedIPAddressSerializer(required=False, allow_null=True)
  137. nat_outside = NestedIPAddressSerializer(many=True, read_only=True)
  138. class Meta:
  139. model = IPAddress
  140. fields = [
  141. 'id', 'url', 'display', 'family', 'address', 'vrf', 'tenant', 'status', 'role', 'assigned_object_type',
  142. 'assigned_object_id', 'assigned_object', 'nat_inside', 'nat_outside', 'dns_name', 'description', 'comments',
  143. 'tags', 'custom_fields', 'created', 'last_updated',
  144. ]
  145. brief_fields = ('id', 'url', 'display', 'family', 'address', 'description')
  146. @extend_schema_field(serializers.JSONField(allow_null=True))
  147. def get_assigned_object(self, obj):
  148. if obj.assigned_object is None:
  149. return None
  150. serializer = get_serializer_for_model(obj.assigned_object)
  151. context = {'request': self.context['request']}
  152. return serializer(obj.assigned_object, nested=True, context=context).data
  153. class AvailableIPSerializer(serializers.Serializer):
  154. """
  155. Representation of an IP address which does not exist in the database.
  156. """
  157. family = serializers.IntegerField(read_only=True)
  158. address = serializers.CharField(read_only=True)
  159. vrf = VRFSerializer(nested=True, read_only=True, allow_null=True)
  160. description = serializers.CharField(required=False)
  161. def to_representation(self, instance):
  162. if self.context.get('vrf'):
  163. vrf = VRFSerializer(self.context['vrf'], nested=True, context={'request': self.context['request']}).data
  164. else:
  165. vrf = None
  166. return {
  167. 'family': self.context['parent'].family,
  168. 'address': f"{instance}/{self.context['parent'].mask_length}",
  169. 'vrf': vrf,
  170. }