serializers.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320
  1. from __future__ import unicode_literals
  2. from collections import OrderedDict
  3. from rest_framework import serializers
  4. from rest_framework.reverse import reverse
  5. from rest_framework.validators import UniqueTogetherValidator
  6. from taggit.models import Tag
  7. from dcim.api.serializers import NestedDeviceSerializer, InterfaceSerializer, NestedSiteSerializer
  8. from dcim.models import Interface
  9. from extras.api.customfields import CustomFieldModelSerializer
  10. from ipam.constants import (
  11. IPADDRESS_ROLE_CHOICES, IPADDRESS_STATUS_CHOICES, IP_PROTOCOL_CHOICES, PREFIX_STATUS_CHOICES, VLAN_STATUS_CHOICES,
  12. )
  13. from ipam.models import Aggregate, IPAddress, Prefix, RIR, Role, Service, VLAN, VLANGroup, VRF
  14. from tenancy.api.serializers import NestedTenantSerializer
  15. from utilities.api import (
  16. ChoiceField, SerializedPKRelatedField, TagField, ValidatedModelSerializer, WritableNestedSerializer,
  17. )
  18. from virtualization.api.serializers import NestedVirtualMachineSerializer
  19. #
  20. # VRFs
  21. #
  22. class VRFSerializer(CustomFieldModelSerializer):
  23. tenant = NestedTenantSerializer(required=False, allow_null=True)
  24. tags = TagField(queryset=Tag.objects.all(), required=False, many=True)
  25. class Meta:
  26. model = VRF
  27. fields = [
  28. 'id', 'name', 'rd', 'tenant', 'enforce_unique', 'description', 'tags', 'display_name', 'custom_fields',
  29. 'created', 'last_updated',
  30. ]
  31. class NestedVRFSerializer(WritableNestedSerializer):
  32. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:vrf-detail')
  33. class Meta:
  34. model = VRF
  35. fields = ['id', 'url', 'name', 'rd']
  36. #
  37. # Roles
  38. #
  39. class RoleSerializer(ValidatedModelSerializer):
  40. class Meta:
  41. model = Role
  42. fields = ['id', 'name', 'slug', 'weight']
  43. class NestedRoleSerializer(WritableNestedSerializer):
  44. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:role-detail')
  45. class Meta:
  46. model = Role
  47. fields = ['id', 'url', 'name', 'slug']
  48. #
  49. # RIRs
  50. #
  51. class RIRSerializer(ValidatedModelSerializer):
  52. class Meta:
  53. model = RIR
  54. fields = ['id', 'name', 'slug', 'is_private']
  55. class NestedRIRSerializer(WritableNestedSerializer):
  56. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:rir-detail')
  57. class Meta:
  58. model = RIR
  59. fields = ['id', 'url', 'name', 'slug']
  60. #
  61. # Aggregates
  62. #
  63. class AggregateSerializer(CustomFieldModelSerializer):
  64. rir = NestedRIRSerializer()
  65. tags = TagField(queryset=Tag.objects.all(), required=False, many=True)
  66. class Meta:
  67. model = Aggregate
  68. fields = [
  69. 'id', 'family', 'prefix', 'rir', 'date_added', 'description', 'tags', 'custom_fields', 'created',
  70. 'last_updated',
  71. ]
  72. read_only_fields = ['family']
  73. class NestedAggregateSerializer(WritableNestedSerializer):
  74. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:aggregate-detail')
  75. class Meta(AggregateSerializer.Meta):
  76. model = Aggregate
  77. fields = ['id', 'url', 'family', 'prefix']
  78. #
  79. # VLAN groups
  80. #
  81. class VLANGroupSerializer(ValidatedModelSerializer):
  82. site = NestedSiteSerializer(required=False, allow_null=True)
  83. class Meta:
  84. model = VLANGroup
  85. fields = ['id', 'name', 'slug', 'site']
  86. validators = []
  87. def validate(self, data):
  88. # Validate uniqueness of name and slug if a site has been assigned.
  89. if data.get('site', None):
  90. for field in ['name', 'slug']:
  91. validator = UniqueTogetherValidator(queryset=VLANGroup.objects.all(), fields=('site', field))
  92. validator.set_context(self)
  93. validator(data)
  94. # Enforce model validation
  95. super(VLANGroupSerializer, self).validate(data)
  96. return data
  97. class NestedVLANGroupSerializer(WritableNestedSerializer):
  98. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:vlangroup-detail')
  99. class Meta:
  100. model = VLANGroup
  101. fields = ['id', 'url', 'name', 'slug']
  102. #
  103. # VLANs
  104. #
  105. class VLANSerializer(CustomFieldModelSerializer):
  106. site = NestedSiteSerializer(required=False, allow_null=True)
  107. group = NestedVLANGroupSerializer(required=False, allow_null=True)
  108. tenant = NestedTenantSerializer(required=False, allow_null=True)
  109. status = ChoiceField(choices=VLAN_STATUS_CHOICES, required=False)
  110. role = NestedRoleSerializer(required=False, allow_null=True)
  111. tags = TagField(queryset=Tag.objects.all(), required=False, many=True)
  112. class Meta:
  113. model = VLAN
  114. fields = [
  115. 'id', 'site', 'group', 'vid', 'name', 'tenant', 'status', 'role', 'description', 'tags', 'display_name',
  116. 'custom_fields', 'created', 'last_updated',
  117. ]
  118. validators = []
  119. def validate(self, data):
  120. # Validate uniqueness of vid and name if a group has been assigned.
  121. if data.get('group', None):
  122. for field in ['vid', 'name']:
  123. validator = UniqueTogetherValidator(queryset=VLAN.objects.all(), fields=('group', field))
  124. validator.set_context(self)
  125. validator(data)
  126. # Enforce model validation
  127. super(VLANSerializer, self).validate(data)
  128. return data
  129. class NestedVLANSerializer(WritableNestedSerializer):
  130. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:vlan-detail')
  131. class Meta:
  132. model = VLAN
  133. fields = ['id', 'url', 'vid', 'name', 'display_name']
  134. #
  135. # Prefixes
  136. #
  137. class PrefixSerializer(CustomFieldModelSerializer):
  138. site = NestedSiteSerializer(required=False, allow_null=True)
  139. vrf = NestedVRFSerializer(required=False, allow_null=True)
  140. tenant = NestedTenantSerializer(required=False, allow_null=True)
  141. vlan = NestedVLANSerializer(required=False, allow_null=True)
  142. status = ChoiceField(choices=PREFIX_STATUS_CHOICES, required=False)
  143. role = NestedRoleSerializer(required=False, allow_null=True)
  144. tags = TagField(queryset=Tag.objects.all(), required=False, many=True)
  145. class Meta:
  146. model = Prefix
  147. fields = [
  148. 'id', 'family', 'prefix', 'site', 'vrf', 'tenant', 'vlan', 'status', 'role', 'is_pool', 'description',
  149. 'tags', 'custom_fields', 'created', 'last_updated',
  150. ]
  151. read_only_fields = ['family']
  152. class NestedPrefixSerializer(WritableNestedSerializer):
  153. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:prefix-detail')
  154. class Meta:
  155. model = Prefix
  156. fields = ['id', 'url', 'family', 'prefix']
  157. class AvailablePrefixSerializer(serializers.Serializer):
  158. def to_representation(self, instance):
  159. if self.context.get('vrf'):
  160. vrf = NestedVRFSerializer(self.context['vrf'], context={'request': self.context['request']}).data
  161. else:
  162. vrf = None
  163. return OrderedDict([
  164. ('family', instance.version),
  165. ('prefix', str(instance)),
  166. ('vrf', vrf),
  167. ])
  168. #
  169. # IP addresses
  170. #
  171. class IPAddressInterfaceSerializer(WritableNestedSerializer):
  172. url = serializers.SerializerMethodField() # We're imitating a HyperlinkedIdentityField here
  173. device = NestedDeviceSerializer(read_only=True)
  174. virtual_machine = NestedVirtualMachineSerializer(read_only=True)
  175. class Meta(InterfaceSerializer.Meta):
  176. model = Interface
  177. fields = [
  178. 'id', 'url', 'device', 'virtual_machine', 'name',
  179. ]
  180. def get_url(self, obj):
  181. """
  182. Return a link to the Interface via either the DCIM API if the parent is a Device, or via the virtualization API
  183. if the parent is a VirtualMachine.
  184. """
  185. url_name = 'dcim-api:interface-detail' if obj.device else 'virtualization-api:interface-detail'
  186. return reverse(url_name, kwargs={'pk': obj.pk}, request=self.context['request'])
  187. class IPAddressSerializer(CustomFieldModelSerializer):
  188. vrf = NestedVRFSerializer(required=False, allow_null=True)
  189. tenant = NestedTenantSerializer(required=False, allow_null=True)
  190. status = ChoiceField(choices=IPADDRESS_STATUS_CHOICES, required=False)
  191. role = ChoiceField(choices=IPADDRESS_ROLE_CHOICES, required=False)
  192. interface = IPAddressInterfaceSerializer(required=False, allow_null=True)
  193. tags = TagField(queryset=Tag.objects.all(), required=False, many=True)
  194. class Meta:
  195. model = IPAddress
  196. fields = [
  197. 'id', 'family', 'address', 'vrf', 'tenant', 'status', 'role', 'interface', 'description', 'nat_inside',
  198. 'nat_outside', 'tags', 'custom_fields', 'created', 'last_updated',
  199. ]
  200. read_only_fields = ['family']
  201. class NestedIPAddressSerializer(WritableNestedSerializer):
  202. url = serializers.HyperlinkedIdentityField(view_name='ipam-api:ipaddress-detail')
  203. class Meta:
  204. model = IPAddress
  205. fields = ['id', 'url', 'family', 'address']
  206. IPAddressSerializer._declared_fields['nat_inside'] = NestedIPAddressSerializer(required=False, allow_null=True)
  207. IPAddressSerializer._declared_fields['nat_outside'] = NestedIPAddressSerializer(read_only=True)
  208. class AvailableIPSerializer(serializers.Serializer):
  209. def to_representation(self, instance):
  210. if self.context.get('vrf'):
  211. vrf = NestedVRFSerializer(self.context['vrf'], context={'request': self.context['request']}).data
  212. else:
  213. vrf = None
  214. return OrderedDict([
  215. ('family', self.context['prefix'].version),
  216. ('address', '{}/{}'.format(instance, self.context['prefix'].prefixlen)),
  217. ('vrf', vrf),
  218. ])
  219. #
  220. # Services
  221. #
  222. class ServiceSerializer(CustomFieldModelSerializer):
  223. device = NestedDeviceSerializer(required=False, allow_null=True)
  224. virtual_machine = NestedVirtualMachineSerializer(required=False, allow_null=True)
  225. protocol = ChoiceField(choices=IP_PROTOCOL_CHOICES)
  226. ipaddresses = SerializedPKRelatedField(
  227. queryset=IPAddress.objects.all(),
  228. serializer=NestedIPAddressSerializer,
  229. required=False,
  230. many=True
  231. )
  232. class Meta:
  233. model = Service
  234. fields = [
  235. 'id', 'device', 'virtual_machine', 'name', 'port', 'protocol', 'ipaddresses', 'description',
  236. 'custom_fields', 'created', 'last_updated',
  237. ]