serializers.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  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 dcim.api.nested_serializers import NestedDeviceSerializer, NestedSiteSerializer
  6. from ipam.choices import *
  7. from ipam.constants import IPADDRESS_ASSIGNMENT_MODELS, VLANGROUP_SCOPE_TYPES
  8. from ipam.models import *
  9. from netbox.api import ChoiceField, ContentTypeField, SerializedPKRelatedField
  10. from netbox.api.serializers import PrimaryModelSerializer
  11. from tenancy.api.nested_serializers import NestedTenantSerializer
  12. from utilities.api import get_serializer_for_model
  13. from virtualization.api.nested_serializers import NestedVirtualMachineSerializer
  14. from .nested_serializers import *
  15. #
  16. # ASNs
  17. #
  18. class ASNSerializer(PrimaryModelSerializer):
  19. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:asn-detail')
  20. tenant = NestedTenantSerializer(required=False, allow_null=True)
  21. site_count = serializers.IntegerField(read_only=True)
  22. class Meta:
  23. model = ASN
  24. fields = [
  25. 'id', 'url', 'display', 'asn', 'site_count', 'rir', 'tenant', 'description', 'tags', 'custom_fields',
  26. 'created', 'last_updated',
  27. ]
  28. #
  29. # VRFs
  30. #
  31. class VRFSerializer(PrimaryModelSerializer):
  32. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:vrf-detail')
  33. tenant = NestedTenantSerializer(required=False, allow_null=True)
  34. import_targets = SerializedPKRelatedField(
  35. queryset=RouteTarget.objects.all(),
  36. serializer=NestedRouteTargetSerializer,
  37. required=False,
  38. many=True
  39. )
  40. export_targets = SerializedPKRelatedField(
  41. queryset=RouteTarget.objects.all(),
  42. serializer=NestedRouteTargetSerializer,
  43. required=False,
  44. many=True
  45. )
  46. ipaddress_count = serializers.IntegerField(read_only=True)
  47. prefix_count = serializers.IntegerField(read_only=True)
  48. class Meta:
  49. model = VRF
  50. fields = [
  51. 'id', 'url', 'display', 'name', 'rd', 'tenant', 'enforce_unique', 'description', 'import_targets',
  52. 'export_targets', 'tags', 'custom_fields', 'created', 'last_updated', 'ipaddress_count', 'prefix_count',
  53. ]
  54. #
  55. # Route targets
  56. #
  57. class RouteTargetSerializer(PrimaryModelSerializer):
  58. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:routetarget-detail')
  59. tenant = NestedTenantSerializer(required=False, allow_null=True)
  60. class Meta:
  61. model = RouteTarget
  62. fields = [
  63. 'id', 'url', 'display', 'name', 'tenant', 'description', 'tags', 'custom_fields', 'created', 'last_updated',
  64. ]
  65. #
  66. # RIRs/aggregates
  67. #
  68. class RIRSerializer(PrimaryModelSerializer):
  69. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:rir-detail')
  70. aggregate_count = serializers.IntegerField(read_only=True)
  71. class Meta:
  72. model = RIR
  73. fields = [
  74. 'id', 'url', 'display', 'name', 'slug', 'is_private', 'description', 'tags', 'custom_fields', 'created',
  75. 'last_updated', 'aggregate_count',
  76. ]
  77. class AggregateSerializer(PrimaryModelSerializer):
  78. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:aggregate-detail')
  79. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  80. rir = NestedRIRSerializer()
  81. tenant = NestedTenantSerializer(required=False, allow_null=True)
  82. class Meta:
  83. model = Aggregate
  84. fields = [
  85. 'id', 'url', 'display', 'family', 'prefix', 'rir', 'tenant', 'date_added', 'description', 'tags',
  86. 'custom_fields', 'created', 'last_updated',
  87. ]
  88. read_only_fields = ['family']
  89. #
  90. # FHRP Groups
  91. #
  92. class FHRPGroupSerializer(PrimaryModelSerializer):
  93. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:fhrpgroup-detail')
  94. ip_addresses = NestedIPAddressSerializer(many=True, read_only=True)
  95. class Meta:
  96. model = FHRPGroup
  97. fields = [
  98. 'id', 'url', 'display', 'protocol', 'group_id', 'auth_type', 'auth_key', 'description', 'ip_addresses',
  99. 'tags', 'custom_fields', 'created', 'last_updated',
  100. ]
  101. class FHRPGroupAssignmentSerializer(PrimaryModelSerializer):
  102. url = serializers.HyperlinkedIdentityField(view_name='tenancy-api:contactassignment-detail')
  103. group = NestedFHRPGroupSerializer()
  104. interface_type = ContentTypeField(
  105. queryset=ContentType.objects.all()
  106. )
  107. interface = serializers.SerializerMethodField(read_only=True)
  108. class Meta:
  109. model = FHRPGroupAssignment
  110. fields = [
  111. 'id', 'url', 'display', 'group', 'interface_type', 'interface_id', 'interface', 'priority', 'created',
  112. 'last_updated',
  113. ]
  114. @swagger_serializer_method(serializer_or_field=serializers.DictField)
  115. def get_interface(self, obj):
  116. if obj.interface is None:
  117. return None
  118. serializer = get_serializer_for_model(obj.interface, prefix='Nested')
  119. context = {'request': self.context['request']}
  120. return serializer(obj.interface, context=context).data
  121. #
  122. # VLANs
  123. #
  124. class RoleSerializer(PrimaryModelSerializer):
  125. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:role-detail')
  126. prefix_count = serializers.IntegerField(read_only=True)
  127. vlan_count = serializers.IntegerField(read_only=True)
  128. class Meta:
  129. model = Role
  130. fields = [
  131. 'id', 'url', 'display', 'name', 'slug', 'weight', 'description', 'tags', 'custom_fields', 'created',
  132. 'last_updated', 'prefix_count', 'vlan_count',
  133. ]
  134. class VLANGroupSerializer(PrimaryModelSerializer):
  135. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:vlangroup-detail')
  136. scope_type = ContentTypeField(
  137. queryset=ContentType.objects.filter(
  138. model__in=VLANGROUP_SCOPE_TYPES
  139. ),
  140. required=False,
  141. default=None
  142. )
  143. scope_id = serializers.IntegerField(allow_null=True, required=False, default=None)
  144. scope = serializers.SerializerMethodField(read_only=True)
  145. vlan_count = serializers.IntegerField(read_only=True)
  146. class Meta:
  147. model = VLANGroup
  148. fields = [
  149. 'id', 'url', 'display', 'name', 'slug', 'scope_type', 'scope_id', 'scope', 'min_vid', 'max_vid',
  150. 'description', 'tags', 'custom_fields', 'created', 'last_updated', 'vlan_count',
  151. ]
  152. validators = []
  153. def get_scope(self, obj):
  154. if obj.scope_id is None:
  155. return None
  156. serializer = get_serializer_for_model(obj.scope, prefix='Nested')
  157. context = {'request': self.context['request']}
  158. return serializer(obj.scope, context=context).data
  159. class VLANSerializer(PrimaryModelSerializer):
  160. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:vlan-detail')
  161. site = NestedSiteSerializer(required=False, allow_null=True)
  162. group = NestedVLANGroupSerializer(required=False, allow_null=True, default=None)
  163. tenant = NestedTenantSerializer(required=False, allow_null=True)
  164. status = ChoiceField(choices=VLANStatusChoices, required=False)
  165. role = NestedRoleSerializer(required=False, allow_null=True)
  166. prefix_count = serializers.IntegerField(read_only=True)
  167. class Meta:
  168. model = VLAN
  169. fields = [
  170. 'id', 'url', 'display', 'site', 'group', 'vid', 'name', 'tenant', 'status', 'role', 'description', 'tags',
  171. 'custom_fields', 'created', 'last_updated', 'prefix_count',
  172. ]
  173. class AvailableVLANSerializer(serializers.Serializer):
  174. """
  175. Representation of a VLAN which does not exist in the database.
  176. """
  177. vid = serializers.IntegerField(read_only=True)
  178. group = NestedVLANGroupSerializer(read_only=True)
  179. def to_representation(self, instance):
  180. return OrderedDict([
  181. ('vid', instance),
  182. ('group', NestedVLANGroupSerializer(
  183. self.context['group'],
  184. context={'request': self.context['request']}
  185. ).data),
  186. ])
  187. class CreateAvailableVLANSerializer(PrimaryModelSerializer):
  188. site = NestedSiteSerializer(required=False, allow_null=True)
  189. tenant = NestedTenantSerializer(required=False, allow_null=True)
  190. status = ChoiceField(choices=VLANStatusChoices, required=False)
  191. role = NestedRoleSerializer(required=False, allow_null=True)
  192. class Meta:
  193. model = VLAN
  194. fields = [
  195. 'name', 'site', 'tenant', 'status', 'role', 'description', 'tags', 'custom_fields',
  196. ]
  197. def validate(self, data):
  198. # Bypass model validation since we don't have a VID yet
  199. return data
  200. #
  201. # Prefixes
  202. #
  203. class PrefixSerializer(PrimaryModelSerializer):
  204. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:prefix-detail')
  205. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  206. site = NestedSiteSerializer(required=False, allow_null=True)
  207. vrf = NestedVRFSerializer(required=False, allow_null=True)
  208. tenant = NestedTenantSerializer(required=False, allow_null=True)
  209. vlan = NestedVLANSerializer(required=False, allow_null=True)
  210. status = ChoiceField(choices=PrefixStatusChoices, required=False)
  211. role = NestedRoleSerializer(required=False, allow_null=True)
  212. children = serializers.IntegerField(read_only=True)
  213. _depth = serializers.IntegerField(read_only=True)
  214. class Meta:
  215. model = Prefix
  216. fields = [
  217. 'id', 'url', 'display', 'family', 'prefix', 'site', 'vrf', 'tenant', 'vlan', 'status', 'role', 'is_pool',
  218. 'mark_utilized', 'description', 'tags', 'custom_fields', 'created', 'last_updated', 'children', '_depth',
  219. ]
  220. read_only_fields = ['family']
  221. class PrefixLengthSerializer(serializers.Serializer):
  222. prefix_length = serializers.IntegerField()
  223. def to_internal_value(self, data):
  224. requested_prefix = data.get('prefix_length')
  225. if requested_prefix is None:
  226. raise serializers.ValidationError({
  227. 'prefix_length': 'this field can not be missing'
  228. })
  229. if not isinstance(requested_prefix, int):
  230. raise serializers.ValidationError({
  231. 'prefix_length': 'this field must be int type'
  232. })
  233. prefix = self.context.get('prefix')
  234. if prefix.family == 4 and requested_prefix > 32:
  235. raise serializers.ValidationError({
  236. 'prefix_length': 'Invalid prefix length ({}) for IPv4'.format((requested_prefix))
  237. })
  238. elif prefix.family == 6 and requested_prefix > 128:
  239. raise serializers.ValidationError({
  240. 'prefix_length': 'Invalid prefix length ({}) for IPv6'.format((requested_prefix))
  241. })
  242. return data
  243. class AvailablePrefixSerializer(serializers.Serializer):
  244. """
  245. Representation of a prefix which does not exist in the database.
  246. """
  247. family = serializers.IntegerField(read_only=True)
  248. prefix = serializers.CharField(read_only=True)
  249. vrf = NestedVRFSerializer(read_only=True)
  250. def to_representation(self, instance):
  251. if self.context.get('vrf'):
  252. vrf = NestedVRFSerializer(self.context['vrf'], context={'request': self.context['request']}).data
  253. else:
  254. vrf = None
  255. return OrderedDict([
  256. ('family', instance.version),
  257. ('prefix', str(instance)),
  258. ('vrf', vrf),
  259. ])
  260. #
  261. # IP ranges
  262. #
  263. class IPRangeSerializer(PrimaryModelSerializer):
  264. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:iprange-detail')
  265. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  266. vrf = NestedVRFSerializer(required=False, allow_null=True)
  267. tenant = NestedTenantSerializer(required=False, allow_null=True)
  268. status = ChoiceField(choices=IPRangeStatusChoices, required=False)
  269. role = NestedRoleSerializer(required=False, allow_null=True)
  270. children = serializers.IntegerField(read_only=True)
  271. class Meta:
  272. model = IPRange
  273. fields = [
  274. 'id', 'url', 'display', 'family', 'start_address', 'end_address', 'size', 'vrf', 'tenant', 'status', 'role',
  275. 'description', 'tags', 'custom_fields', 'created', 'last_updated', 'children',
  276. ]
  277. read_only_fields = ['family']
  278. #
  279. # IP addresses
  280. #
  281. class IPAddressSerializer(PrimaryModelSerializer):
  282. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:ipaddress-detail')
  283. family = ChoiceField(choices=IPAddressFamilyChoices, read_only=True)
  284. vrf = NestedVRFSerializer(required=False, allow_null=True)
  285. tenant = NestedTenantSerializer(required=False, allow_null=True)
  286. status = ChoiceField(choices=IPAddressStatusChoices, required=False)
  287. role = ChoiceField(choices=IPAddressRoleChoices, allow_blank=True, required=False)
  288. assigned_object_type = ContentTypeField(
  289. queryset=ContentType.objects.filter(IPADDRESS_ASSIGNMENT_MODELS),
  290. required=False,
  291. allow_null=True
  292. )
  293. assigned_object = serializers.SerializerMethodField(read_only=True)
  294. nat_inside = NestedIPAddressSerializer(required=False, allow_null=True)
  295. nat_outside = NestedIPAddressSerializer(required=False, read_only=True)
  296. class Meta:
  297. model = IPAddress
  298. fields = [
  299. 'id', 'url', 'display', 'family', 'address', 'vrf', 'tenant', 'status', 'role', 'assigned_object_type',
  300. 'assigned_object_id', 'assigned_object', 'nat_inside', 'nat_outside', 'dns_name', 'description', 'tags',
  301. 'custom_fields', 'created', 'last_updated',
  302. ]
  303. read_only_fields = ['family', 'nat_outside']
  304. @swagger_serializer_method(serializer_or_field=serializers.DictField)
  305. def get_assigned_object(self, obj):
  306. if obj.assigned_object is None:
  307. return None
  308. serializer = get_serializer_for_model(obj.assigned_object, prefix='Nested')
  309. context = {'request': self.context['request']}
  310. return serializer(obj.assigned_object, context=context).data
  311. class AvailableIPSerializer(serializers.Serializer):
  312. """
  313. Representation of an IP address which does not exist in the database.
  314. """
  315. family = serializers.IntegerField(read_only=True)
  316. address = serializers.CharField(read_only=True)
  317. vrf = NestedVRFSerializer(read_only=True)
  318. def to_representation(self, instance):
  319. if self.context.get('vrf'):
  320. vrf = NestedVRFSerializer(self.context['vrf'], context={'request': self.context['request']}).data
  321. else:
  322. vrf = None
  323. return OrderedDict([
  324. ('family', self.context['parent'].family),
  325. ('address', f"{instance}/{self.context['parent'].mask_length}"),
  326. ('vrf', vrf),
  327. ])
  328. #
  329. # Services
  330. #
  331. class ServiceSerializer(PrimaryModelSerializer):
  332. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:service-detail')
  333. device = NestedDeviceSerializer(required=False, allow_null=True)
  334. virtual_machine = NestedVirtualMachineSerializer(required=False, allow_null=True)
  335. protocol = ChoiceField(choices=ServiceProtocolChoices, required=False)
  336. ipaddresses = SerializedPKRelatedField(
  337. queryset=IPAddress.objects.all(),
  338. serializer=NestedIPAddressSerializer,
  339. required=False,
  340. many=True
  341. )
  342. class Meta:
  343. model = Service
  344. fields = [
  345. 'id', 'url', 'display', 'device', 'virtual_machine', 'name', 'ports', 'protocol', 'ipaddresses',
  346. 'description', 'tags', 'custom_fields', 'created', 'last_updated',
  347. ]