serializers.py 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065
  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 timezone_field.rest_framework import TimeZoneSerializerField
  5. from dcim.choices import *
  6. from dcim.constants import *
  7. from dcim.models import *
  8. from ipam.api.nested_serializers import (
  9. NestedASNSerializer, NestedIPAddressSerializer, NestedVLANSerializer, NestedVRFSerializer,
  10. )
  11. from ipam.models import ASN, VLAN
  12. from netbox.api import ChoiceField, ContentTypeField, SerializedPKRelatedField
  13. from netbox.api.serializers import (
  14. NestedGroupModelSerializer, PrimaryModelSerializer, ValidatedModelSerializer, WritableNestedSerializer,
  15. )
  16. from netbox.config import ConfigItem
  17. from tenancy.api.nested_serializers import NestedTenantSerializer
  18. from users.api.nested_serializers import NestedUserSerializer
  19. from utilities.api import get_serializer_for_model
  20. from virtualization.api.nested_serializers import NestedClusterSerializer
  21. from wireless.api.nested_serializers import NestedWirelessLANSerializer, NestedWirelessLinkSerializer
  22. from wireless.choices import *
  23. from wireless.models import WirelessLAN
  24. from .nested_serializers import *
  25. class LinkTerminationSerializer(serializers.ModelSerializer):
  26. link_peer_type = serializers.SerializerMethodField(read_only=True)
  27. link_peer = serializers.SerializerMethodField(read_only=True)
  28. _occupied = serializers.SerializerMethodField(read_only=True)
  29. def get_link_peer_type(self, obj):
  30. if obj._link_peer is not None:
  31. return f'{obj._link_peer._meta.app_label}.{obj._link_peer._meta.model_name}'
  32. return None
  33. @swagger_serializer_method(serializer_or_field=serializers.DictField)
  34. def get_link_peer(self, obj):
  35. """
  36. Return the appropriate serializer for the link termination model.
  37. """
  38. if obj._link_peer is not None:
  39. serializer = get_serializer_for_model(obj._link_peer, prefix='Nested')
  40. context = {'request': self.context['request']}
  41. return serializer(obj._link_peer, context=context).data
  42. return None
  43. @swagger_serializer_method(serializer_or_field=serializers.BooleanField)
  44. def get__occupied(self, obj):
  45. return obj._occupied
  46. class ConnectedEndpointSerializer(serializers.ModelSerializer):
  47. connected_endpoint_type = serializers.SerializerMethodField(read_only=True)
  48. connected_endpoint = serializers.SerializerMethodField(read_only=True)
  49. connected_endpoint_reachable = serializers.SerializerMethodField(read_only=True)
  50. def get_connected_endpoint_type(self, obj):
  51. if obj._path is not None and obj._path.destination is not None:
  52. return f'{obj._path.destination._meta.app_label}.{obj._path.destination._meta.model_name}'
  53. return None
  54. @swagger_serializer_method(serializer_or_field=serializers.DictField)
  55. def get_connected_endpoint(self, obj):
  56. """
  57. Return the appropriate serializer for the type of connected object.
  58. """
  59. if obj._path is not None and obj._path.destination is not None:
  60. serializer = get_serializer_for_model(obj._path.destination, prefix='Nested')
  61. context = {'request': self.context['request']}
  62. return serializer(obj._path.destination, context=context).data
  63. return None
  64. @swagger_serializer_method(serializer_or_field=serializers.BooleanField)
  65. def get_connected_endpoint_reachable(self, obj):
  66. if obj._path is not None:
  67. return obj._path.is_active
  68. return None
  69. #
  70. # Regions/sites
  71. #
  72. class RegionSerializer(NestedGroupModelSerializer):
  73. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:region-detail')
  74. parent = NestedRegionSerializer(required=False, allow_null=True, default=None)
  75. site_count = serializers.IntegerField(read_only=True)
  76. class Meta:
  77. model = Region
  78. fields = [
  79. 'id', 'url', 'display', 'name', 'slug', 'parent', 'description', 'tags', 'custom_fields', 'created',
  80. 'last_updated', 'site_count', '_depth',
  81. ]
  82. class SiteGroupSerializer(NestedGroupModelSerializer):
  83. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:sitegroup-detail')
  84. parent = NestedSiteGroupSerializer(required=False, allow_null=True, default=None)
  85. site_count = serializers.IntegerField(read_only=True)
  86. class Meta:
  87. model = SiteGroup
  88. fields = [
  89. 'id', 'url', 'display', 'name', 'slug', 'parent', 'description', 'tags', 'custom_fields', 'created',
  90. 'last_updated', 'site_count', '_depth',
  91. ]
  92. class SiteSerializer(PrimaryModelSerializer):
  93. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:site-detail')
  94. status = ChoiceField(choices=SiteStatusChoices, required=False)
  95. region = NestedRegionSerializer(required=False, allow_null=True)
  96. group = NestedSiteGroupSerializer(required=False, allow_null=True)
  97. tenant = NestedTenantSerializer(required=False, allow_null=True)
  98. time_zone = TimeZoneSerializerField(required=False)
  99. asns = SerializedPKRelatedField(
  100. queryset=ASN.objects.all(),
  101. serializer=NestedASNSerializer,
  102. required=False,
  103. many=True
  104. )
  105. # Related object counts
  106. circuit_count = serializers.IntegerField(read_only=True)
  107. device_count = serializers.IntegerField(read_only=True)
  108. prefix_count = serializers.IntegerField(read_only=True)
  109. rack_count = serializers.IntegerField(read_only=True)
  110. virtualmachine_count = serializers.IntegerField(read_only=True)
  111. vlan_count = serializers.IntegerField(read_only=True)
  112. class Meta:
  113. model = Site
  114. fields = [
  115. 'id', 'url', 'display', 'name', 'slug', 'status', 'region', 'group', 'tenant', 'facility', 'asns',
  116. 'time_zone', 'description', 'physical_address', 'shipping_address', 'latitude', 'longitude', 'comments',
  117. 'tags', 'custom_fields', 'created', 'last_updated', 'circuit_count', 'device_count', 'prefix_count',
  118. 'rack_count', 'virtualmachine_count', 'vlan_count',
  119. ]
  120. #
  121. # Racks
  122. #
  123. class LocationSerializer(NestedGroupModelSerializer):
  124. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:location-detail')
  125. site = NestedSiteSerializer()
  126. parent = NestedLocationSerializer(required=False, allow_null=True)
  127. tenant = NestedTenantSerializer(required=False, allow_null=True)
  128. rack_count = serializers.IntegerField(read_only=True)
  129. device_count = serializers.IntegerField(read_only=True)
  130. class Meta:
  131. model = Location
  132. fields = [
  133. 'id', 'url', 'display', 'name', 'slug', 'site', 'parent', 'tenant', 'description', 'tags', 'custom_fields',
  134. 'created', 'last_updated', 'rack_count', 'device_count', '_depth',
  135. ]
  136. class RackRoleSerializer(PrimaryModelSerializer):
  137. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:rackrole-detail')
  138. rack_count = serializers.IntegerField(read_only=True)
  139. class Meta:
  140. model = RackRole
  141. fields = [
  142. 'id', 'url', 'display', 'name', 'slug', 'color', 'description', 'tags', 'custom_fields', 'created',
  143. 'last_updated', 'rack_count',
  144. ]
  145. class RackSerializer(PrimaryModelSerializer):
  146. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:rack-detail')
  147. site = NestedSiteSerializer()
  148. location = NestedLocationSerializer(required=False, allow_null=True, default=None)
  149. tenant = NestedTenantSerializer(required=False, allow_null=True)
  150. status = ChoiceField(choices=RackStatusChoices, required=False)
  151. role = NestedRackRoleSerializer(required=False, allow_null=True)
  152. type = ChoiceField(choices=RackTypeChoices, allow_blank=True, required=False)
  153. facility_id = serializers.CharField(max_length=50, allow_blank=True, allow_null=True, label='Facility ID',
  154. default=None)
  155. width = ChoiceField(choices=RackWidthChoices, required=False)
  156. outer_unit = ChoiceField(choices=RackDimensionUnitChoices, allow_blank=True, required=False)
  157. device_count = serializers.IntegerField(read_only=True)
  158. powerfeed_count = serializers.IntegerField(read_only=True)
  159. class Meta:
  160. model = Rack
  161. fields = [
  162. 'id', 'url', 'display', 'name', 'facility_id', 'site', 'location', 'tenant', 'status', 'role', 'serial',
  163. 'asset_tag', 'type', 'width', 'u_height', 'desc_units', 'outer_width', 'outer_depth', 'outer_unit',
  164. 'comments', 'tags', 'custom_fields', 'created', 'last_updated', 'device_count', 'powerfeed_count',
  165. ]
  166. class RackUnitSerializer(serializers.Serializer):
  167. """
  168. A rack unit is an abstraction formed by the set (rack, position, face); it does not exist as a row in the database.
  169. """
  170. id = serializers.IntegerField(read_only=True)
  171. name = serializers.CharField(read_only=True)
  172. face = ChoiceField(choices=DeviceFaceChoices, read_only=True)
  173. device = NestedDeviceSerializer(read_only=True)
  174. occupied = serializers.BooleanField(read_only=True)
  175. display = serializers.SerializerMethodField(read_only=True)
  176. def get_display(self, obj):
  177. return obj['name']
  178. class RackReservationSerializer(PrimaryModelSerializer):
  179. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:rackreservation-detail')
  180. rack = NestedRackSerializer()
  181. user = NestedUserSerializer()
  182. tenant = NestedTenantSerializer(required=False, allow_null=True)
  183. class Meta:
  184. model = RackReservation
  185. fields = [
  186. 'id', 'url', 'display', 'rack', 'units', 'created', 'user', 'tenant', 'description', 'tags',
  187. 'custom_fields',
  188. ]
  189. class RackElevationDetailFilterSerializer(serializers.Serializer):
  190. q = serializers.CharField(
  191. required=False,
  192. default=None
  193. )
  194. face = serializers.ChoiceField(
  195. choices=DeviceFaceChoices,
  196. default=DeviceFaceChoices.FACE_FRONT
  197. )
  198. render = serializers.ChoiceField(
  199. choices=RackElevationDetailRenderChoices,
  200. default=RackElevationDetailRenderChoices.RENDER_JSON
  201. )
  202. unit_width = serializers.IntegerField(
  203. default=ConfigItem('RACK_ELEVATION_DEFAULT_UNIT_WIDTH')
  204. )
  205. unit_height = serializers.IntegerField(
  206. default=ConfigItem('RACK_ELEVATION_DEFAULT_UNIT_HEIGHT')
  207. )
  208. legend_width = serializers.IntegerField(
  209. default=RACK_ELEVATION_LEGEND_WIDTH_DEFAULT
  210. )
  211. exclude = serializers.IntegerField(
  212. required=False,
  213. default=None
  214. )
  215. expand_devices = serializers.BooleanField(
  216. required=False,
  217. default=True
  218. )
  219. include_images = serializers.BooleanField(
  220. required=False,
  221. default=True
  222. )
  223. #
  224. # Device/module types
  225. #
  226. class ManufacturerSerializer(PrimaryModelSerializer):
  227. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:manufacturer-detail')
  228. devicetype_count = serializers.IntegerField(read_only=True)
  229. inventoryitem_count = serializers.IntegerField(read_only=True)
  230. platform_count = serializers.IntegerField(read_only=True)
  231. class Meta:
  232. model = Manufacturer
  233. fields = [
  234. 'id', 'url', 'display', 'name', 'slug', 'description', 'tags', 'custom_fields', 'created', 'last_updated',
  235. 'devicetype_count', 'inventoryitem_count', 'platform_count',
  236. ]
  237. class DeviceTypeSerializer(PrimaryModelSerializer):
  238. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:devicetype-detail')
  239. manufacturer = NestedManufacturerSerializer()
  240. subdevice_role = ChoiceField(choices=SubdeviceRoleChoices, allow_blank=True, required=False)
  241. airflow = ChoiceField(choices=DeviceAirflowChoices, allow_blank=True, required=False)
  242. device_count = serializers.IntegerField(read_only=True)
  243. class Meta:
  244. model = DeviceType
  245. fields = [
  246. 'id', 'url', 'display', 'manufacturer', 'model', 'slug', 'part_number', 'u_height', 'is_full_depth',
  247. 'subdevice_role', 'airflow', 'front_image', 'rear_image', 'comments', 'tags', 'custom_fields', 'created',
  248. 'last_updated', 'device_count',
  249. ]
  250. class ModuleTypeSerializer(PrimaryModelSerializer):
  251. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:moduletype-detail')
  252. manufacturer = NestedManufacturerSerializer()
  253. # module_count = serializers.IntegerField(read_only=True)
  254. class Meta:
  255. model = ModuleType
  256. fields = [
  257. 'id', 'url', 'display', 'manufacturer', 'model', 'part_number', 'comments', 'tags', 'custom_fields',
  258. 'created', 'last_updated',
  259. ]
  260. #
  261. # Component templates
  262. #
  263. class ConsolePortTemplateSerializer(ValidatedModelSerializer):
  264. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:consoleporttemplate-detail')
  265. device_type = NestedDeviceTypeSerializer()
  266. type = ChoiceField(
  267. choices=ConsolePortTypeChoices,
  268. allow_blank=True,
  269. required=False
  270. )
  271. class Meta:
  272. model = ConsolePortTemplate
  273. fields = [
  274. 'id', 'url', 'display', 'device_type', 'name', 'label', 'type', 'description', 'created', 'last_updated',
  275. ]
  276. class ConsoleServerPortTemplateSerializer(ValidatedModelSerializer):
  277. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:consoleserverporttemplate-detail')
  278. device_type = NestedDeviceTypeSerializer()
  279. type = ChoiceField(
  280. choices=ConsolePortTypeChoices,
  281. allow_blank=True,
  282. required=False
  283. )
  284. class Meta:
  285. model = ConsoleServerPortTemplate
  286. fields = [
  287. 'id', 'url', 'display', 'device_type', 'name', 'label', 'type', 'description', 'created', 'last_updated',
  288. ]
  289. class PowerPortTemplateSerializer(ValidatedModelSerializer):
  290. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:powerporttemplate-detail')
  291. device_type = NestedDeviceTypeSerializer()
  292. type = ChoiceField(
  293. choices=PowerPortTypeChoices,
  294. allow_blank=True,
  295. required=False
  296. )
  297. class Meta:
  298. model = PowerPortTemplate
  299. fields = [
  300. 'id', 'url', 'display', 'device_type', 'name', 'label', 'type', 'maximum_draw', 'allocated_draw',
  301. 'description', 'created', 'last_updated',
  302. ]
  303. class PowerOutletTemplateSerializer(ValidatedModelSerializer):
  304. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:poweroutlettemplate-detail')
  305. device_type = NestedDeviceTypeSerializer()
  306. type = ChoiceField(
  307. choices=PowerOutletTypeChoices,
  308. allow_blank=True,
  309. required=False
  310. )
  311. power_port = NestedPowerPortTemplateSerializer(
  312. required=False,
  313. allow_null=True
  314. )
  315. feed_leg = ChoiceField(
  316. choices=PowerOutletFeedLegChoices,
  317. allow_blank=True,
  318. required=False
  319. )
  320. class Meta:
  321. model = PowerOutletTemplate
  322. fields = [
  323. 'id', 'url', 'display', 'device_type', 'name', 'label', 'type', 'power_port', 'feed_leg', 'description',
  324. 'created', 'last_updated',
  325. ]
  326. class InterfaceTemplateSerializer(ValidatedModelSerializer):
  327. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:interfacetemplate-detail')
  328. device_type = NestedDeviceTypeSerializer()
  329. type = ChoiceField(choices=InterfaceTypeChoices)
  330. class Meta:
  331. model = InterfaceTemplate
  332. fields = [
  333. 'id', 'url', 'display', 'device_type', 'name', 'label', 'type', 'mgmt_only', 'description', 'created',
  334. 'last_updated',
  335. ]
  336. class RearPortTemplateSerializer(ValidatedModelSerializer):
  337. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:rearporttemplate-detail')
  338. device_type = NestedDeviceTypeSerializer()
  339. type = ChoiceField(choices=PortTypeChoices)
  340. class Meta:
  341. model = RearPortTemplate
  342. fields = [
  343. 'id', 'url', 'display', 'device_type', 'name', 'label', 'type', 'color', 'positions', 'description',
  344. 'created', 'last_updated',
  345. ]
  346. class FrontPortTemplateSerializer(ValidatedModelSerializer):
  347. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:frontporttemplate-detail')
  348. device_type = NestedDeviceTypeSerializer()
  349. type = ChoiceField(choices=PortTypeChoices)
  350. rear_port = NestedRearPortTemplateSerializer()
  351. class Meta:
  352. model = FrontPortTemplate
  353. fields = [
  354. 'id', 'url', 'display', 'device_type', 'name', 'label', 'type', 'color', 'rear_port', 'rear_port_position',
  355. 'description', 'created', 'last_updated',
  356. ]
  357. class ModuleBayTemplateSerializer(ValidatedModelSerializer):
  358. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:modulebaytemplate-detail')
  359. device_type = NestedDeviceTypeSerializer()
  360. class Meta:
  361. model = ModuleBayTemplate
  362. fields = [
  363. 'id', 'url', 'display', 'device_type', 'name', 'label', 'position', 'description', 'created',
  364. 'last_updated',
  365. ]
  366. class DeviceBayTemplateSerializer(ValidatedModelSerializer):
  367. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:devicebaytemplate-detail')
  368. device_type = NestedDeviceTypeSerializer()
  369. class Meta:
  370. model = DeviceBayTemplate
  371. fields = ['id', 'url', 'display', 'device_type', 'name', 'label', 'description', 'created', 'last_updated']
  372. class InventoryItemTemplateSerializer(ValidatedModelSerializer):
  373. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:inventoryitemtemplate-detail')
  374. device_type = NestedDeviceTypeSerializer()
  375. parent = serializers.PrimaryKeyRelatedField(
  376. queryset=InventoryItemTemplate.objects.all(),
  377. allow_null=True,
  378. default=None
  379. )
  380. role = NestedInventoryItemRoleSerializer(required=False, allow_null=True)
  381. manufacturer = NestedManufacturerSerializer(required=False, allow_null=True, default=None)
  382. component_type = ContentTypeField(
  383. queryset=ContentType.objects.filter(MODULAR_COMPONENT_TEMPLATE_MODELS),
  384. required=False,
  385. allow_null=True
  386. )
  387. component = serializers.SerializerMethodField(read_only=True)
  388. _depth = serializers.IntegerField(source='level', read_only=True)
  389. class Meta:
  390. model = InventoryItemTemplate
  391. fields = [
  392. 'id', 'url', 'display', 'device_type', 'parent', 'name', 'label', 'role', 'manufacturer', 'part_id',
  393. 'description', 'component_type', 'component_id', 'component', 'created', 'last_updated', '_depth',
  394. ]
  395. @swagger_serializer_method(serializer_or_field=serializers.DictField)
  396. def get_component(self, obj):
  397. if obj.component is None:
  398. return None
  399. serializer = get_serializer_for_model(obj.component, prefix='Nested')
  400. context = {'request': self.context['request']}
  401. return serializer(obj.component, context=context).data
  402. #
  403. # Devices
  404. #
  405. class DeviceRoleSerializer(PrimaryModelSerializer):
  406. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:devicerole-detail')
  407. device_count = serializers.IntegerField(read_only=True)
  408. virtualmachine_count = serializers.IntegerField(read_only=True)
  409. class Meta:
  410. model = DeviceRole
  411. fields = [
  412. 'id', 'url', 'display', 'name', 'slug', 'color', 'vm_role', 'description', 'tags', 'custom_fields',
  413. 'created', 'last_updated', 'device_count', 'virtualmachine_count',
  414. ]
  415. class PlatformSerializer(PrimaryModelSerializer):
  416. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:platform-detail')
  417. manufacturer = NestedManufacturerSerializer(required=False, allow_null=True)
  418. device_count = serializers.IntegerField(read_only=True)
  419. virtualmachine_count = serializers.IntegerField(read_only=True)
  420. class Meta:
  421. model = Platform
  422. fields = [
  423. 'id', 'url', 'display', 'name', 'slug', 'manufacturer', 'napalm_driver', 'napalm_args', 'description',
  424. 'tags', 'custom_fields', 'created', 'last_updated', 'device_count', 'virtualmachine_count',
  425. ]
  426. class DeviceSerializer(PrimaryModelSerializer):
  427. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:device-detail')
  428. device_type = NestedDeviceTypeSerializer()
  429. device_role = NestedDeviceRoleSerializer()
  430. tenant = NestedTenantSerializer(required=False, allow_null=True, default=None)
  431. platform = NestedPlatformSerializer(required=False, allow_null=True)
  432. site = NestedSiteSerializer()
  433. location = NestedLocationSerializer(required=False, allow_null=True, default=None)
  434. rack = NestedRackSerializer(required=False, allow_null=True, default=None)
  435. face = ChoiceField(choices=DeviceFaceChoices, allow_blank=True, default='')
  436. position = serializers.IntegerField(allow_null=True, label='Position (U)', min_value=1, default=None)
  437. status = ChoiceField(choices=DeviceStatusChoices, required=False)
  438. airflow = ChoiceField(choices=DeviceAirflowChoices, allow_blank=True, required=False)
  439. primary_ip = NestedIPAddressSerializer(read_only=True)
  440. primary_ip4 = NestedIPAddressSerializer(required=False, allow_null=True)
  441. primary_ip6 = NestedIPAddressSerializer(required=False, allow_null=True)
  442. parent_device = serializers.SerializerMethodField()
  443. cluster = NestedClusterSerializer(required=False, allow_null=True)
  444. virtual_chassis = NestedVirtualChassisSerializer(required=False, allow_null=True, default=None)
  445. vc_position = serializers.IntegerField(allow_null=True, max_value=255, min_value=0, default=None)
  446. class Meta:
  447. model = Device
  448. fields = [
  449. 'id', 'url', 'display', 'name', 'device_type', 'device_role', 'tenant', 'platform', 'serial', 'asset_tag',
  450. 'site', 'location', 'rack', 'position', 'face', 'parent_device', 'status', 'airflow', 'primary_ip',
  451. 'primary_ip4', 'primary_ip6', 'cluster', 'virtual_chassis', 'vc_position', 'vc_priority', 'comments',
  452. 'local_context_data', 'tags', 'custom_fields', 'created', 'last_updated',
  453. ]
  454. @swagger_serializer_method(serializer_or_field=NestedDeviceSerializer)
  455. def get_parent_device(self, obj):
  456. try:
  457. device_bay = obj.parent_bay
  458. except DeviceBay.DoesNotExist:
  459. return None
  460. context = {'request': self.context['request']}
  461. data = NestedDeviceSerializer(instance=device_bay.device, context=context).data
  462. data['device_bay'] = NestedDeviceBaySerializer(instance=device_bay, context=context).data
  463. return data
  464. class ModuleSerializer(PrimaryModelSerializer):
  465. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:module-detail')
  466. device = NestedDeviceSerializer()
  467. module_bay = NestedModuleBaySerializer()
  468. module_type = NestedModuleTypeSerializer()
  469. class Meta:
  470. model = Module
  471. fields = [
  472. 'id', 'url', 'display', 'device', 'module_bay', 'module_type', 'serial', 'asset_tag', 'comments', 'tags',
  473. 'custom_fields', 'created', 'last_updated',
  474. ]
  475. class DeviceWithConfigContextSerializer(DeviceSerializer):
  476. config_context = serializers.SerializerMethodField()
  477. class Meta(DeviceSerializer.Meta):
  478. fields = [
  479. 'id', 'url', 'display', 'name', 'device_type', 'device_role', 'tenant', 'platform', 'serial', 'asset_tag',
  480. 'site', 'location', 'rack', 'position', 'face', 'parent_device', 'status', 'primary_ip', 'primary_ip4',
  481. 'primary_ip6', 'cluster', 'virtual_chassis', 'vc_position', 'vc_priority', 'comments', 'local_context_data',
  482. 'tags', 'custom_fields', 'config_context', 'created', 'last_updated',
  483. ]
  484. @swagger_serializer_method(serializer_or_field=serializers.DictField)
  485. def get_config_context(self, obj):
  486. return obj.get_config_context()
  487. class DeviceNAPALMSerializer(serializers.Serializer):
  488. method = serializers.DictField()
  489. #
  490. # Device components
  491. #
  492. class ConsoleServerPortSerializer(PrimaryModelSerializer, LinkTerminationSerializer, ConnectedEndpointSerializer):
  493. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:consoleserverport-detail')
  494. device = NestedDeviceSerializer()
  495. module = ComponentNestedModuleSerializer(
  496. required=False,
  497. allow_null=True
  498. )
  499. type = ChoiceField(
  500. choices=ConsolePortTypeChoices,
  501. allow_blank=True,
  502. required=False
  503. )
  504. speed = ChoiceField(
  505. choices=ConsolePortSpeedChoices,
  506. allow_null=True,
  507. required=False
  508. )
  509. cable = NestedCableSerializer(read_only=True)
  510. class Meta:
  511. model = ConsoleServerPort
  512. fields = [
  513. 'id', 'url', 'display', 'device', 'module', 'name', 'label', 'type', 'speed', 'description',
  514. 'mark_connected', 'cable', 'link_peer', 'link_peer_type', 'connected_endpoint', 'connected_endpoint_type',
  515. 'connected_endpoint_reachable', 'tags', 'custom_fields', 'created', 'last_updated', '_occupied',
  516. ]
  517. class ConsolePortSerializer(PrimaryModelSerializer, LinkTerminationSerializer, ConnectedEndpointSerializer):
  518. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:consoleport-detail')
  519. device = NestedDeviceSerializer()
  520. module = ComponentNestedModuleSerializer(
  521. required=False,
  522. allow_null=True
  523. )
  524. type = ChoiceField(
  525. choices=ConsolePortTypeChoices,
  526. allow_blank=True,
  527. required=False
  528. )
  529. speed = ChoiceField(
  530. choices=ConsolePortSpeedChoices,
  531. allow_null=True,
  532. required=False
  533. )
  534. cable = NestedCableSerializer(read_only=True)
  535. class Meta:
  536. model = ConsolePort
  537. fields = [
  538. 'id', 'url', 'display', 'device', 'module', 'name', 'label', 'type', 'speed', 'description',
  539. 'mark_connected', 'cable', 'link_peer', 'link_peer_type', 'connected_endpoint', 'connected_endpoint_type',
  540. 'connected_endpoint_reachable', 'tags', 'custom_fields', 'created', 'last_updated', '_occupied',
  541. ]
  542. class PowerOutletSerializer(PrimaryModelSerializer, LinkTerminationSerializer, ConnectedEndpointSerializer):
  543. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:poweroutlet-detail')
  544. device = NestedDeviceSerializer()
  545. module = ComponentNestedModuleSerializer(
  546. required=False,
  547. allow_null=True
  548. )
  549. type = ChoiceField(
  550. choices=PowerOutletTypeChoices,
  551. allow_blank=True,
  552. required=False
  553. )
  554. power_port = NestedPowerPortSerializer(
  555. required=False,
  556. allow_null=True
  557. )
  558. feed_leg = ChoiceField(
  559. choices=PowerOutletFeedLegChoices,
  560. allow_blank=True,
  561. required=False
  562. )
  563. cable = NestedCableSerializer(
  564. read_only=True
  565. )
  566. class Meta:
  567. model = PowerOutlet
  568. fields = [
  569. 'id', 'url', 'display', 'device', 'module', 'name', 'label', 'type', 'power_port', 'feed_leg',
  570. 'description', 'mark_connected', 'cable', 'link_peer', 'link_peer_type', 'connected_endpoint',
  571. 'connected_endpoint_type', 'connected_endpoint_reachable', 'tags', 'custom_fields', 'created',
  572. 'last_updated', '_occupied',
  573. ]
  574. class PowerPortSerializer(PrimaryModelSerializer, LinkTerminationSerializer, ConnectedEndpointSerializer):
  575. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:powerport-detail')
  576. device = NestedDeviceSerializer()
  577. module = ComponentNestedModuleSerializer(
  578. required=False,
  579. allow_null=True
  580. )
  581. type = ChoiceField(
  582. choices=PowerPortTypeChoices,
  583. allow_blank=True,
  584. required=False
  585. )
  586. cable = NestedCableSerializer(read_only=True)
  587. class Meta:
  588. model = PowerPort
  589. fields = [
  590. 'id', 'url', 'display', 'device', 'module', 'name', 'label', 'type', 'maximum_draw', 'allocated_draw',
  591. 'description', 'mark_connected', 'cable', 'link_peer', 'link_peer_type', 'connected_endpoint',
  592. 'connected_endpoint_type', 'connected_endpoint_reachable', 'tags', 'custom_fields', 'created',
  593. 'last_updated', '_occupied',
  594. ]
  595. class InterfaceSerializer(PrimaryModelSerializer, LinkTerminationSerializer, ConnectedEndpointSerializer):
  596. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:interface-detail')
  597. device = NestedDeviceSerializer()
  598. module = ComponentNestedModuleSerializer(
  599. required=False,
  600. allow_null=True
  601. )
  602. type = ChoiceField(choices=InterfaceTypeChoices)
  603. parent = NestedInterfaceSerializer(required=False, allow_null=True)
  604. bridge = NestedInterfaceSerializer(required=False, allow_null=True)
  605. lag = NestedInterfaceSerializer(required=False, allow_null=True)
  606. mode = ChoiceField(choices=InterfaceModeChoices, allow_blank=True, required=False)
  607. rf_role = ChoiceField(choices=WirelessRoleChoices, required=False, allow_null=True)
  608. rf_channel = ChoiceField(choices=WirelessChannelChoices, required=False)
  609. untagged_vlan = NestedVLANSerializer(required=False, allow_null=True)
  610. tagged_vlans = SerializedPKRelatedField(
  611. queryset=VLAN.objects.all(),
  612. serializer=NestedVLANSerializer,
  613. required=False,
  614. many=True
  615. )
  616. vrf = NestedVRFSerializer(required=False, allow_null=True)
  617. cable = NestedCableSerializer(read_only=True)
  618. wireless_link = NestedWirelessLinkSerializer(read_only=True)
  619. wireless_lans = SerializedPKRelatedField(
  620. queryset=WirelessLAN.objects.all(),
  621. serializer=NestedWirelessLANSerializer,
  622. required=False,
  623. many=True
  624. )
  625. count_ipaddresses = serializers.IntegerField(read_only=True)
  626. count_fhrp_groups = serializers.IntegerField(read_only=True)
  627. class Meta:
  628. model = Interface
  629. fields = [
  630. 'id', 'url', 'display', 'device', 'module', 'name', 'label', 'type', 'enabled', 'parent', 'bridge', 'lag',
  631. 'mtu', 'mac_address', 'wwn', 'mgmt_only', 'description', 'mode', 'rf_role', 'rf_channel',
  632. 'rf_channel_frequency', 'rf_channel_width', 'tx_power', 'untagged_vlan', 'tagged_vlans', 'mark_connected',
  633. 'cable', 'wireless_link', 'link_peer', 'link_peer_type', 'wireless_lans', 'vrf', 'connected_endpoint',
  634. 'connected_endpoint_type', 'connected_endpoint_reachable', 'tags', 'custom_fields', 'created',
  635. 'last_updated', 'count_ipaddresses', 'count_fhrp_groups', '_occupied',
  636. ]
  637. def validate(self, data):
  638. # Validate many-to-many VLAN assignments
  639. device = self.instance.device if self.instance else data.get('device')
  640. for vlan in data.get('tagged_vlans', []):
  641. if vlan.site not in [device.site, None]:
  642. raise serializers.ValidationError({
  643. 'tagged_vlans': f"VLAN {vlan} must belong to the same site as the interface's parent device, or "
  644. f"it must be global."
  645. })
  646. return super().validate(data)
  647. class RearPortSerializer(PrimaryModelSerializer, LinkTerminationSerializer):
  648. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:rearport-detail')
  649. device = NestedDeviceSerializer()
  650. module = ComponentNestedModuleSerializer(
  651. required=False,
  652. allow_null=True
  653. )
  654. type = ChoiceField(choices=PortTypeChoices)
  655. cable = NestedCableSerializer(read_only=True)
  656. class Meta:
  657. model = RearPort
  658. fields = [
  659. 'id', 'url', 'display', 'device', 'module', 'name', 'label', 'type', 'color', 'positions', 'description',
  660. 'mark_connected', 'cable', 'link_peer', 'link_peer_type', 'tags', 'custom_fields', 'created',
  661. 'last_updated', '_occupied',
  662. ]
  663. class FrontPortRearPortSerializer(WritableNestedSerializer):
  664. """
  665. NestedRearPortSerializer but with parent device omitted (since front and rear ports must belong to same device)
  666. """
  667. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:rearport-detail')
  668. class Meta:
  669. model = RearPort
  670. fields = ['id', 'url', 'display', 'name', 'label']
  671. class FrontPortSerializer(PrimaryModelSerializer, LinkTerminationSerializer):
  672. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:frontport-detail')
  673. device = NestedDeviceSerializer()
  674. module = ComponentNestedModuleSerializer(
  675. required=False,
  676. allow_null=True
  677. )
  678. type = ChoiceField(choices=PortTypeChoices)
  679. rear_port = FrontPortRearPortSerializer()
  680. cable = NestedCableSerializer(read_only=True)
  681. class Meta:
  682. model = FrontPort
  683. fields = [
  684. 'id', 'url', 'display', 'device', 'module', 'name', 'label', 'type', 'color', 'rear_port',
  685. 'rear_port_position', 'description', 'mark_connected', 'cable', 'link_peer', 'link_peer_type', 'tags',
  686. 'custom_fields', 'created', 'last_updated', '_occupied',
  687. ]
  688. class ModuleBaySerializer(PrimaryModelSerializer):
  689. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:modulebay-detail')
  690. device = NestedDeviceSerializer()
  691. # installed_module = NestedModuleSerializer(required=False, allow_null=True)
  692. class Meta:
  693. model = ModuleBay
  694. fields = [
  695. 'id', 'url', 'display', 'device', 'name', 'label', 'position', 'description', 'tags', 'custom_fields',
  696. 'created', 'last_updated',
  697. ]
  698. class DeviceBaySerializer(PrimaryModelSerializer):
  699. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:devicebay-detail')
  700. device = NestedDeviceSerializer()
  701. installed_device = NestedDeviceSerializer(required=False, allow_null=True)
  702. class Meta:
  703. model = DeviceBay
  704. fields = [
  705. 'id', 'url', 'display', 'device', 'name', 'label', 'description', 'installed_device', 'tags',
  706. 'custom_fields', 'created', 'last_updated',
  707. ]
  708. class InventoryItemSerializer(PrimaryModelSerializer):
  709. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:inventoryitem-detail')
  710. device = NestedDeviceSerializer()
  711. parent = serializers.PrimaryKeyRelatedField(queryset=InventoryItem.objects.all(), allow_null=True, default=None)
  712. role = NestedInventoryItemRoleSerializer(required=False, allow_null=True)
  713. manufacturer = NestedManufacturerSerializer(required=False, allow_null=True, default=None)
  714. component_type = ContentTypeField(
  715. queryset=ContentType.objects.filter(MODULAR_COMPONENT_MODELS),
  716. required=False,
  717. allow_null=True
  718. )
  719. component = serializers.SerializerMethodField(read_only=True)
  720. _depth = serializers.IntegerField(source='level', read_only=True)
  721. class Meta:
  722. model = InventoryItem
  723. fields = [
  724. 'id', 'url', 'display', 'device', 'parent', 'name', 'label', 'role', 'manufacturer', 'part_id', 'serial',
  725. 'asset_tag', 'discovered', 'description', 'component_type', 'component_id', 'component', 'tags',
  726. 'custom_fields', 'created', 'last_updated', '_depth',
  727. ]
  728. @swagger_serializer_method(serializer_or_field=serializers.DictField)
  729. def get_component(self, obj):
  730. if obj.component is None:
  731. return None
  732. serializer = get_serializer_for_model(obj.component, prefix='Nested')
  733. context = {'request': self.context['request']}
  734. return serializer(obj.component, context=context).data
  735. #
  736. # Device component roles
  737. #
  738. class InventoryItemRoleSerializer(PrimaryModelSerializer):
  739. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:inventoryitemrole-detail')
  740. inventoryitem_count = serializers.IntegerField(read_only=True)
  741. class Meta:
  742. model = InventoryItemRole
  743. fields = [
  744. 'id', 'url', 'display', 'name', 'slug', 'color', 'description', 'tags', 'custom_fields', 'created',
  745. 'last_updated', 'inventoryitem_count',
  746. ]
  747. #
  748. # Cables
  749. #
  750. class CableSerializer(PrimaryModelSerializer):
  751. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:cable-detail')
  752. termination_a_type = ContentTypeField(
  753. queryset=ContentType.objects.filter(CABLE_TERMINATION_MODELS)
  754. )
  755. termination_b_type = ContentTypeField(
  756. queryset=ContentType.objects.filter(CABLE_TERMINATION_MODELS)
  757. )
  758. termination_a = serializers.SerializerMethodField(read_only=True)
  759. termination_b = serializers.SerializerMethodField(read_only=True)
  760. status = ChoiceField(choices=LinkStatusChoices, required=False)
  761. tenant = NestedTenantSerializer(required=False, allow_null=True)
  762. length_unit = ChoiceField(choices=CableLengthUnitChoices, allow_blank=True, required=False)
  763. class Meta:
  764. model = Cable
  765. fields = [
  766. 'id', 'url', 'display', 'termination_a_type', 'termination_a_id', 'termination_a', 'termination_b_type',
  767. 'termination_b_id', 'termination_b', 'type', 'status', 'tenant', 'label', 'color', 'length', 'length_unit',
  768. 'tags', 'custom_fields',
  769. ]
  770. def _get_termination(self, obj, side):
  771. """
  772. Serialize a nested representation of a termination.
  773. """
  774. if side.lower() not in ['a', 'b']:
  775. raise ValueError("Termination side must be either A or B.")
  776. termination = getattr(obj, 'termination_{}'.format(side.lower()))
  777. if termination is None:
  778. return None
  779. serializer = get_serializer_for_model(termination, prefix='Nested')
  780. context = {'request': self.context['request']}
  781. data = serializer(termination, context=context).data
  782. return data
  783. @swagger_serializer_method(serializer_or_field=serializers.DictField)
  784. def get_termination_a(self, obj):
  785. return self._get_termination(obj, 'a')
  786. @swagger_serializer_method(serializer_or_field=serializers.DictField)
  787. def get_termination_b(self, obj):
  788. return self._get_termination(obj, 'b')
  789. class TracedCableSerializer(serializers.ModelSerializer):
  790. """
  791. Used only while tracing a cable path.
  792. """
  793. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:cable-detail')
  794. class Meta:
  795. model = Cable
  796. fields = [
  797. 'id', 'url', 'type', 'status', 'label', 'color', 'length', 'length_unit',
  798. ]
  799. class CablePathSerializer(serializers.ModelSerializer):
  800. origin_type = ContentTypeField(read_only=True)
  801. origin = serializers.SerializerMethodField(read_only=True)
  802. destination_type = ContentTypeField(read_only=True)
  803. destination = serializers.SerializerMethodField(read_only=True)
  804. path = serializers.SerializerMethodField(read_only=True)
  805. class Meta:
  806. model = CablePath
  807. fields = [
  808. 'id', 'origin_type', 'origin', 'destination_type', 'destination', 'path', 'is_active', 'is_split',
  809. ]
  810. @swagger_serializer_method(serializer_or_field=serializers.DictField)
  811. def get_origin(self, obj):
  812. """
  813. Return the appropriate serializer for the origin.
  814. """
  815. serializer = get_serializer_for_model(obj.origin, prefix='Nested')
  816. context = {'request': self.context['request']}
  817. return serializer(obj.origin, context=context).data
  818. @swagger_serializer_method(serializer_or_field=serializers.DictField)
  819. def get_destination(self, obj):
  820. """
  821. Return the appropriate serializer for the destination, if any.
  822. """
  823. if obj.destination_id is not None:
  824. serializer = get_serializer_for_model(obj.destination, prefix='Nested')
  825. context = {'request': self.context['request']}
  826. return serializer(obj.destination, context=context).data
  827. return None
  828. @swagger_serializer_method(serializer_or_field=serializers.ListField)
  829. def get_path(self, obj):
  830. ret = []
  831. for node in obj.get_path():
  832. serializer = get_serializer_for_model(node, prefix='Nested')
  833. context = {'request': self.context['request']}
  834. ret.append(serializer(node, context=context).data)
  835. return ret
  836. #
  837. # Virtual chassis
  838. #
  839. class VirtualChassisSerializer(PrimaryModelSerializer):
  840. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:virtualchassis-detail')
  841. master = NestedDeviceSerializer(required=False)
  842. member_count = serializers.IntegerField(read_only=True)
  843. class Meta:
  844. model = VirtualChassis
  845. fields = ['id', 'url', 'display', 'name', 'domain', 'master', 'tags', 'custom_fields', 'member_count']
  846. #
  847. # Power panels
  848. #
  849. class PowerPanelSerializer(PrimaryModelSerializer):
  850. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:powerpanel-detail')
  851. site = NestedSiteSerializer()
  852. location = NestedLocationSerializer(
  853. required=False,
  854. allow_null=True,
  855. default=None
  856. )
  857. powerfeed_count = serializers.IntegerField(read_only=True)
  858. class Meta:
  859. model = PowerPanel
  860. fields = ['id', 'url', 'display', 'site', 'location', 'name', 'tags', 'custom_fields', 'powerfeed_count']
  861. class PowerFeedSerializer(PrimaryModelSerializer, LinkTerminationSerializer, ConnectedEndpointSerializer):
  862. url = serializers.HyperlinkedIdentityField(view_name='dcim-api:powerfeed-detail')
  863. power_panel = NestedPowerPanelSerializer()
  864. rack = NestedRackSerializer(
  865. required=False,
  866. allow_null=True,
  867. default=None
  868. )
  869. type = ChoiceField(
  870. choices=PowerFeedTypeChoices,
  871. default=PowerFeedTypeChoices.TYPE_PRIMARY
  872. )
  873. status = ChoiceField(
  874. choices=PowerFeedStatusChoices,
  875. default=PowerFeedStatusChoices.STATUS_ACTIVE
  876. )
  877. supply = ChoiceField(
  878. choices=PowerFeedSupplyChoices,
  879. default=PowerFeedSupplyChoices.SUPPLY_AC
  880. )
  881. phase = ChoiceField(
  882. choices=PowerFeedPhaseChoices,
  883. default=PowerFeedPhaseChoices.PHASE_SINGLE
  884. )
  885. cable = NestedCableSerializer(read_only=True)
  886. class Meta:
  887. model = PowerFeed
  888. fields = [
  889. 'id', 'url', 'display', 'power_panel', 'rack', 'name', 'status', 'type', 'supply', 'phase', 'voltage',
  890. 'amperage', 'max_utilization', 'comments', 'mark_connected', 'cable', 'link_peer', 'link_peer_type',
  891. 'connected_endpoint', 'connected_endpoint_type', 'connected_endpoint_reachable', 'tags', 'custom_fields',
  892. 'created', 'last_updated', '_occupied',
  893. ]