ip.py 8.9 KB

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