serializers.py 18 KB

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