serializers.py 19 KB

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