serializers.py 19 KB

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