serializers.py 8.6 KB

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