serializers.py 16 KB

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