views.py 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692
  1. from collections import OrderedDict
  2. from django.conf import settings
  3. from django.db.models import Count, F
  4. from django.http import HttpResponseForbidden, HttpResponse
  5. from django.shortcuts import get_object_or_404
  6. from drf_yasg import openapi
  7. from drf_yasg.openapi import Parameter
  8. from drf_yasg.utils import swagger_auto_schema
  9. from rest_framework.decorators import action
  10. from rest_framework.mixins import ListModelMixin
  11. from rest_framework.response import Response
  12. from rest_framework.viewsets import GenericViewSet, ViewSet
  13. from circuits.models import Circuit
  14. from dcim import filters
  15. from dcim.models import (
  16. Cable, ConsolePort, ConsolePortTemplate, ConsoleServerPort, ConsoleServerPortTemplate, Device, DeviceBay,
  17. DeviceBayTemplate, DeviceRole, DeviceType, FrontPort, FrontPortTemplate, Interface, InterfaceTemplate,
  18. Manufacturer, InventoryItem, Platform, PowerFeed, PowerOutlet, PowerOutletTemplate, PowerPanel, PowerPort,
  19. PowerPortTemplate, Rack, RackGroup, RackReservation, RackRole, RearPort, RearPortTemplate, Region, Site,
  20. VirtualChassis,
  21. )
  22. from extras.api.serializers import RenderedGraphSerializer
  23. from extras.api.views import CustomFieldModelViewSet
  24. from extras.models import Graph
  25. from ipam.models import Prefix, VLAN
  26. from utilities.api import (
  27. get_serializer_for_model, IsAuthenticatedOrLoginNotRequired, FieldChoicesViewSet, ModelViewSet, ServiceUnavailable,
  28. )
  29. from utilities.utils import get_subquery
  30. from virtualization.models import VirtualMachine
  31. from . import serializers
  32. from .exceptions import MissingFilterException
  33. #
  34. # Field choices
  35. #
  36. class DCIMFieldChoicesViewSet(FieldChoicesViewSet):
  37. fields = (
  38. (serializers.CableSerializer, ['length_unit', 'status', 'termination_a_type', 'termination_b_type', 'type']),
  39. (serializers.ConsolePortSerializer, ['type', 'connection_status']),
  40. (serializers.ConsolePortTemplateSerializer, ['type']),
  41. (serializers.ConsoleServerPortSerializer, ['type']),
  42. (serializers.ConsoleServerPortTemplateSerializer, ['type']),
  43. (serializers.DeviceSerializer, ['face', 'status']),
  44. (serializers.DeviceTypeSerializer, ['subdevice_role']),
  45. (serializers.FrontPortSerializer, ['type']),
  46. (serializers.FrontPortTemplateSerializer, ['type']),
  47. (serializers.InterfaceSerializer, ['type', 'mode']),
  48. (serializers.InterfaceTemplateSerializer, ['type']),
  49. (serializers.PowerFeedSerializer, ['phase', 'status', 'supply', 'type']),
  50. (serializers.PowerOutletSerializer, ['type', 'feed_leg']),
  51. (serializers.PowerOutletTemplateSerializer, ['type', 'feed_leg']),
  52. (serializers.PowerPortSerializer, ['type', 'connection_status']),
  53. (serializers.PowerPortTemplateSerializer, ['type']),
  54. (serializers.RackSerializer, ['outer_unit', 'status', 'type', 'width']),
  55. (serializers.RearPortSerializer, ['type']),
  56. (serializers.RearPortTemplateSerializer, ['type']),
  57. (serializers.SiteSerializer, ['status']),
  58. )
  59. # Mixins
  60. class CableTraceMixin(object):
  61. @action(detail=True, url_path='trace')
  62. def trace(self, request, pk):
  63. """
  64. Trace a complete cable path and return each segment as a three-tuple of (termination, cable, termination).
  65. """
  66. obj = get_object_or_404(self.queryset.model, pk=pk)
  67. # Initialize the path array
  68. path = []
  69. for near_end, cable, far_end in obj.trace(follow_circuits=True):
  70. # Serialize each object
  71. serializer_a = get_serializer_for_model(near_end, prefix='Nested')
  72. x = serializer_a(near_end, context={'request': request}).data
  73. if cable is not None:
  74. y = serializers.TracedCableSerializer(cable, context={'request': request}).data
  75. else:
  76. y = None
  77. if far_end is not None:
  78. serializer_b = get_serializer_for_model(far_end, prefix='Nested')
  79. z = serializer_b(far_end, context={'request': request}).data
  80. else:
  81. z = None
  82. path.append((x, y, z))
  83. return Response(path)
  84. #
  85. # Regions
  86. #
  87. class RegionViewSet(ModelViewSet):
  88. queryset = Region.objects.annotate(
  89. site_count=Count('sites')
  90. )
  91. serializer_class = serializers.RegionSerializer
  92. filterset_class = filters.RegionFilterSet
  93. #
  94. # Sites
  95. #
  96. class SiteViewSet(CustomFieldModelViewSet):
  97. queryset = Site.objects.prefetch_related(
  98. 'region', 'tenant', 'tags'
  99. ).annotate(
  100. device_count=get_subquery(Device, 'site'),
  101. rack_count=get_subquery(Rack, 'site'),
  102. prefix_count=get_subquery(Prefix, 'site'),
  103. vlan_count=get_subquery(VLAN, 'site'),
  104. circuit_count=get_subquery(Circuit, 'terminations__site'),
  105. virtualmachine_count=get_subquery(VirtualMachine, 'cluster__site'),
  106. )
  107. serializer_class = serializers.SiteSerializer
  108. filterset_class = filters.SiteFilterSet
  109. @action(detail=True)
  110. def graphs(self, request, pk):
  111. """
  112. A convenience method for rendering graphs for a particular site.
  113. """
  114. site = get_object_or_404(Site, pk=pk)
  115. queryset = Graph.objects.filter(type__model='site')
  116. serializer = RenderedGraphSerializer(queryset, many=True, context={'graphed_object': site})
  117. return Response(serializer.data)
  118. #
  119. # Rack groups
  120. #
  121. class RackGroupViewSet(ModelViewSet):
  122. queryset = RackGroup.objects.prefetch_related('site').annotate(
  123. rack_count=Count('racks')
  124. )
  125. serializer_class = serializers.RackGroupSerializer
  126. filterset_class = filters.RackGroupFilterSet
  127. #
  128. # Rack roles
  129. #
  130. class RackRoleViewSet(ModelViewSet):
  131. queryset = RackRole.objects.annotate(
  132. rack_count=Count('racks')
  133. )
  134. serializer_class = serializers.RackRoleSerializer
  135. filterset_class = filters.RackRoleFilterSet
  136. #
  137. # Racks
  138. #
  139. class RackViewSet(CustomFieldModelViewSet):
  140. queryset = Rack.objects.prefetch_related(
  141. 'site', 'group__site', 'role', 'tenant', 'tags'
  142. ).annotate(
  143. device_count=get_subquery(Device, 'rack'),
  144. powerfeed_count=get_subquery(PowerFeed, 'rack')
  145. )
  146. serializer_class = serializers.RackSerializer
  147. filterset_class = filters.RackFilterSet
  148. @swagger_auto_schema(
  149. responses={200: serializers.RackUnitSerializer(many=True)},
  150. query_serializer=serializers.RackElevationDetailFilterSerializer
  151. )
  152. @action(detail=True)
  153. def elevation(self, request, pk=None):
  154. """
  155. Rack elevation representing the list of rack units. Also supports rendering the elevation as an SVG.
  156. """
  157. rack = get_object_or_404(Rack, pk=pk)
  158. serializer = serializers.RackElevationDetailFilterSerializer(data=request.GET)
  159. if not serializer.is_valid():
  160. return Response(serializer.errors, 400)
  161. data = serializer.validated_data
  162. if data['render'] == 'svg':
  163. # Render and return the elevation as an SVG drawing with the correct content type
  164. drawing = rack.get_elevation_svg(
  165. face=data['face'],
  166. unit_width=data['unit_width'],
  167. unit_height=data['unit_height'],
  168. legend_width=data['legend_width'],
  169. include_images=data['include_images']
  170. )
  171. return HttpResponse(drawing.tostring(), content_type='image/svg+xml')
  172. else:
  173. # Return a JSON representation of the rack units in the elevation
  174. elevation = rack.get_rack_units(
  175. face=data['face'],
  176. exclude=data['exclude'],
  177. expand_devices=data['expand_devices']
  178. )
  179. # Enable filtering rack units by ID
  180. q = data['q']
  181. if q:
  182. elevation = [u for u in elevation if q in str(u['id']) or q in str(u['name'])]
  183. page = self.paginate_queryset(elevation)
  184. if page is not None:
  185. rack_units = serializers.RackUnitSerializer(page, many=True, context={'request': request})
  186. return self.get_paginated_response(rack_units.data)
  187. #
  188. # Rack reservations
  189. #
  190. class RackReservationViewSet(ModelViewSet):
  191. queryset = RackReservation.objects.prefetch_related('rack', 'user', 'tenant')
  192. serializer_class = serializers.RackReservationSerializer
  193. filterset_class = filters.RackReservationFilterSet
  194. # Assign user from request
  195. def perform_create(self, serializer):
  196. serializer.save(user=self.request.user)
  197. #
  198. # Manufacturers
  199. #
  200. class ManufacturerViewSet(ModelViewSet):
  201. queryset = Manufacturer.objects.annotate(
  202. devicetype_count=get_subquery(DeviceType, 'manufacturer'),
  203. inventoryitem_count=get_subquery(InventoryItem, 'manufacturer'),
  204. platform_count=get_subquery(Platform, 'manufacturer')
  205. )
  206. serializer_class = serializers.ManufacturerSerializer
  207. filterset_class = filters.ManufacturerFilterSet
  208. #
  209. # Device types
  210. #
  211. class DeviceTypeViewSet(CustomFieldModelViewSet):
  212. queryset = DeviceType.objects.prefetch_related('manufacturer').prefetch_related('tags').annotate(
  213. device_count=Count('instances')
  214. )
  215. serializer_class = serializers.DeviceTypeSerializer
  216. filterset_class = filters.DeviceTypeFilterSet
  217. #
  218. # Device type components
  219. #
  220. class ConsolePortTemplateViewSet(ModelViewSet):
  221. queryset = ConsolePortTemplate.objects.prefetch_related('device_type__manufacturer')
  222. serializer_class = serializers.ConsolePortTemplateSerializer
  223. filterset_class = filters.ConsolePortTemplateFilterSet
  224. class ConsoleServerPortTemplateViewSet(ModelViewSet):
  225. queryset = ConsoleServerPortTemplate.objects.prefetch_related('device_type__manufacturer')
  226. serializer_class = serializers.ConsoleServerPortTemplateSerializer
  227. filterset_class = filters.ConsoleServerPortTemplateFilterSet
  228. class PowerPortTemplateViewSet(ModelViewSet):
  229. queryset = PowerPortTemplate.objects.prefetch_related('device_type__manufacturer')
  230. serializer_class = serializers.PowerPortTemplateSerializer
  231. filterset_class = filters.PowerPortTemplateFilterSet
  232. class PowerOutletTemplateViewSet(ModelViewSet):
  233. queryset = PowerOutletTemplate.objects.prefetch_related('device_type__manufacturer')
  234. serializer_class = serializers.PowerOutletTemplateSerializer
  235. filterset_class = filters.PowerOutletTemplateFilterSet
  236. class InterfaceTemplateViewSet(ModelViewSet):
  237. queryset = InterfaceTemplate.objects.prefetch_related('device_type__manufacturer')
  238. serializer_class = serializers.InterfaceTemplateSerializer
  239. filterset_class = filters.InterfaceTemplateFilterSet
  240. class FrontPortTemplateViewSet(ModelViewSet):
  241. queryset = FrontPortTemplate.objects.prefetch_related('device_type__manufacturer')
  242. serializer_class = serializers.FrontPortTemplateSerializer
  243. filterset_class = filters.FrontPortTemplateFilterSet
  244. class RearPortTemplateViewSet(ModelViewSet):
  245. queryset = RearPortTemplate.objects.prefetch_related('device_type__manufacturer')
  246. serializer_class = serializers.RearPortTemplateSerializer
  247. filterset_class = filters.RearPortTemplateFilterSet
  248. class DeviceBayTemplateViewSet(ModelViewSet):
  249. queryset = DeviceBayTemplate.objects.prefetch_related('device_type__manufacturer')
  250. serializer_class = serializers.DeviceBayTemplateSerializer
  251. filterset_class = filters.DeviceBayTemplateFilterSet
  252. #
  253. # Device roles
  254. #
  255. class DeviceRoleViewSet(ModelViewSet):
  256. queryset = DeviceRole.objects.annotate(
  257. device_count=get_subquery(Device, 'device_role'),
  258. virtualmachine_count=get_subquery(VirtualMachine, 'role')
  259. )
  260. serializer_class = serializers.DeviceRoleSerializer
  261. filterset_class = filters.DeviceRoleFilterSet
  262. #
  263. # Platforms
  264. #
  265. class PlatformViewSet(ModelViewSet):
  266. queryset = Platform.objects.annotate(
  267. device_count=get_subquery(Device, 'platform'),
  268. virtualmachine_count=get_subquery(VirtualMachine, 'platform')
  269. )
  270. serializer_class = serializers.PlatformSerializer
  271. filterset_class = filters.PlatformFilterSet
  272. #
  273. # Devices
  274. #
  275. class DeviceViewSet(CustomFieldModelViewSet):
  276. queryset = Device.objects.prefetch_related(
  277. 'device_type__manufacturer', 'device_role', 'tenant', 'platform', 'site', 'rack', 'parent_bay',
  278. 'virtual_chassis__master', 'primary_ip4__nat_outside', 'primary_ip6__nat_outside', 'tags',
  279. )
  280. filterset_class = filters.DeviceFilterSet
  281. def get_serializer_class(self):
  282. """
  283. Select the specific serializer based on the request context.
  284. If the `brief` query param equates to True, return the NestedDeviceSerializer
  285. If the `exclude` query param includes `config_context` as a value, return the DeviceSerializer
  286. Else, return the DeviceWithConfigContextSerializer
  287. """
  288. request = self.get_serializer_context()['request']
  289. if request.query_params.get('brief', False):
  290. return serializers.NestedDeviceSerializer
  291. elif 'config_context' in request.query_params.get('exclude', []):
  292. return serializers.DeviceSerializer
  293. return serializers.DeviceWithConfigContextSerializer
  294. @action(detail=True)
  295. def graphs(self, request, pk):
  296. """
  297. A convenience method for rendering graphs for a particular Device.
  298. """
  299. device = get_object_or_404(Device, pk=pk)
  300. queryset = Graph.objects.filter(type__model='device')
  301. serializer = RenderedGraphSerializer(queryset, many=True, context={'graphed_object': device})
  302. return Response(serializer.data)
  303. @swagger_auto_schema(
  304. manual_parameters=[
  305. Parameter(
  306. name='method',
  307. in_='query',
  308. required=True,
  309. type=openapi.TYPE_STRING
  310. )
  311. ],
  312. responses={'200': serializers.DeviceNAPALMSerializer}
  313. )
  314. @action(detail=True, url_path='napalm')
  315. def napalm(self, request, pk):
  316. """
  317. Execute a NAPALM method on a Device
  318. """
  319. device = get_object_or_404(Device, pk=pk)
  320. if not device.primary_ip:
  321. raise ServiceUnavailable("This device does not have a primary IP address configured.")
  322. if device.platform is None:
  323. raise ServiceUnavailable("No platform is configured for this device.")
  324. if not device.platform.napalm_driver:
  325. raise ServiceUnavailable("No NAPALM driver is configured for this device's platform ().".format(
  326. device.platform
  327. ))
  328. # Check that NAPALM is installed
  329. try:
  330. import napalm
  331. from napalm.base.exceptions import ModuleImportError
  332. except ImportError:
  333. raise ServiceUnavailable("NAPALM is not installed. Please see the documentation for instructions.")
  334. # Validate the configured driver
  335. try:
  336. driver = napalm.get_network_driver(device.platform.napalm_driver)
  337. except ModuleImportError:
  338. raise ServiceUnavailable("NAPALM driver for platform {} not found: {}.".format(
  339. device.platform, device.platform.napalm_driver
  340. ))
  341. # Verify user permission
  342. if not request.user.has_perm('dcim.napalm_read'):
  343. return HttpResponseForbidden()
  344. # Connect to the device
  345. napalm_methods = request.GET.getlist('method')
  346. response = OrderedDict([(m, None) for m in napalm_methods])
  347. ip_address = str(device.primary_ip.address.ip)
  348. username = settings.NAPALM_USERNAME
  349. password = settings.NAPALM_PASSWORD
  350. optional_args = settings.NAPALM_ARGS.copy()
  351. if device.platform.napalm_args is not None:
  352. optional_args.update(device.platform.napalm_args)
  353. # Update NAPALM parameters according to the request headers
  354. for header in request.headers:
  355. if header[:9].lower() != 'x-napalm-':
  356. continue
  357. key = header[9:]
  358. if key.lower() == 'username':
  359. username = request.headers[header]
  360. elif key.lower() == 'password':
  361. password = request.headers[header]
  362. elif key:
  363. optional_args[key.lower()] = request.headers[header]
  364. d = driver(
  365. hostname=ip_address,
  366. username=username,
  367. password=password,
  368. timeout=settings.NAPALM_TIMEOUT,
  369. optional_args=optional_args
  370. )
  371. try:
  372. d.open()
  373. except Exception as e:
  374. raise ServiceUnavailable("Error connecting to the device at {}: {}".format(ip_address, e))
  375. # Validate and execute each specified NAPALM method
  376. for method in napalm_methods:
  377. if not hasattr(driver, method):
  378. response[method] = {'error': 'Unknown NAPALM method'}
  379. continue
  380. if not method.startswith('get_'):
  381. response[method] = {'error': 'Only get_* NAPALM methods are supported'}
  382. continue
  383. try:
  384. response[method] = getattr(d, method)()
  385. except NotImplementedError:
  386. response[method] = {'error': 'Method {} not implemented for NAPALM driver {}'.format(method, driver)}
  387. except Exception as e:
  388. response[method] = {'error': 'Method {} failed: {}'.format(method, e)}
  389. d.close()
  390. return Response(response)
  391. #
  392. # Device components
  393. #
  394. class ConsolePortViewSet(CableTraceMixin, ModelViewSet):
  395. queryset = ConsolePort.objects.prefetch_related('device', 'connected_endpoint__device', 'cable', 'tags')
  396. serializer_class = serializers.ConsolePortSerializer
  397. filterset_class = filters.ConsolePortFilterSet
  398. class ConsoleServerPortViewSet(CableTraceMixin, ModelViewSet):
  399. queryset = ConsoleServerPort.objects.prefetch_related('device', 'connected_endpoint__device', 'cable', 'tags')
  400. serializer_class = serializers.ConsoleServerPortSerializer
  401. filterset_class = filters.ConsoleServerPortFilterSet
  402. class PowerPortViewSet(CableTraceMixin, ModelViewSet):
  403. queryset = PowerPort.objects.prefetch_related(
  404. 'device', '_connected_poweroutlet__device', '_connected_powerfeed', 'cable', 'tags'
  405. )
  406. serializer_class = serializers.PowerPortSerializer
  407. filterset_class = filters.PowerPortFilterSet
  408. class PowerOutletViewSet(CableTraceMixin, ModelViewSet):
  409. queryset = PowerOutlet.objects.prefetch_related('device', 'connected_endpoint__device', 'cable', 'tags')
  410. serializer_class = serializers.PowerOutletSerializer
  411. filterset_class = filters.PowerOutletFilterSet
  412. class InterfaceViewSet(CableTraceMixin, ModelViewSet):
  413. queryset = Interface.objects.prefetch_related(
  414. 'device', '_connected_interface', '_connected_circuittermination', 'cable', 'ip_addresses', 'tags'
  415. ).filter(
  416. device__isnull=False
  417. )
  418. serializer_class = serializers.InterfaceSerializer
  419. filterset_class = filters.InterfaceFilterSet
  420. @action(detail=True)
  421. def graphs(self, request, pk):
  422. """
  423. A convenience method for rendering graphs for a particular interface.
  424. """
  425. interface = get_object_or_404(Interface, pk=pk)
  426. queryset = Graph.objects.filter(type__model='interface')
  427. serializer = RenderedGraphSerializer(queryset, many=True, context={'graphed_object': interface})
  428. return Response(serializer.data)
  429. class FrontPortViewSet(ModelViewSet):
  430. queryset = FrontPort.objects.prefetch_related('device__device_type__manufacturer', 'rear_port', 'cable', 'tags')
  431. serializer_class = serializers.FrontPortSerializer
  432. filterset_class = filters.FrontPortFilterSet
  433. class RearPortViewSet(ModelViewSet):
  434. queryset = RearPort.objects.prefetch_related('device__device_type__manufacturer', 'cable', 'tags')
  435. serializer_class = serializers.RearPortSerializer
  436. filterset_class = filters.RearPortFilterSet
  437. class DeviceBayViewSet(ModelViewSet):
  438. queryset = DeviceBay.objects.prefetch_related('installed_device').prefetch_related('tags')
  439. serializer_class = serializers.DeviceBaySerializer
  440. filterset_class = filters.DeviceBayFilterSet
  441. class InventoryItemViewSet(ModelViewSet):
  442. queryset = InventoryItem.objects.prefetch_related('device', 'manufacturer').prefetch_related('tags')
  443. serializer_class = serializers.InventoryItemSerializer
  444. filterset_class = filters.InventoryItemFilterSet
  445. #
  446. # Connections
  447. #
  448. class ConsoleConnectionViewSet(ListModelMixin, GenericViewSet):
  449. queryset = ConsolePort.objects.prefetch_related(
  450. 'device', 'connected_endpoint__device'
  451. ).filter(
  452. connected_endpoint__isnull=False
  453. )
  454. serializer_class = serializers.ConsolePortSerializer
  455. filterset_class = filters.ConsoleConnectionFilterSet
  456. class PowerConnectionViewSet(ListModelMixin, GenericViewSet):
  457. queryset = PowerPort.objects.prefetch_related(
  458. 'device', 'connected_endpoint__device'
  459. ).filter(
  460. _connected_poweroutlet__isnull=False
  461. )
  462. serializer_class = serializers.PowerPortSerializer
  463. filterset_class = filters.PowerConnectionFilterSet
  464. class InterfaceConnectionViewSet(ListModelMixin, GenericViewSet):
  465. queryset = Interface.objects.prefetch_related(
  466. 'device', '_connected_interface__device'
  467. ).filter(
  468. # Avoid duplicate connections by only selecting the lower PK in a connected pair
  469. _connected_interface__isnull=False,
  470. pk__lt=F('_connected_interface')
  471. )
  472. serializer_class = serializers.InterfaceConnectionSerializer
  473. filterset_class = filters.InterfaceConnectionFilterSet
  474. #
  475. # Cables
  476. #
  477. class CableViewSet(ModelViewSet):
  478. queryset = Cable.objects.prefetch_related(
  479. 'termination_a', 'termination_b'
  480. )
  481. serializer_class = serializers.CableSerializer
  482. filterset_class = filters.CableFilterSet
  483. #
  484. # Virtual chassis
  485. #
  486. class VirtualChassisViewSet(ModelViewSet):
  487. queryset = VirtualChassis.objects.prefetch_related('tags').annotate(
  488. member_count=Count('members')
  489. )
  490. serializer_class = serializers.VirtualChassisSerializer
  491. filterset_class = filters.VirtualChassisFilterSet
  492. #
  493. # Power panels
  494. #
  495. class PowerPanelViewSet(ModelViewSet):
  496. queryset = PowerPanel.objects.prefetch_related(
  497. 'site', 'rack_group'
  498. ).annotate(
  499. powerfeed_count=Count('powerfeeds')
  500. )
  501. serializer_class = serializers.PowerPanelSerializer
  502. filterset_class = filters.PowerPanelFilterSet
  503. #
  504. # Power feeds
  505. #
  506. class PowerFeedViewSet(CustomFieldModelViewSet):
  507. queryset = PowerFeed.objects.prefetch_related('power_panel', 'rack', 'tags')
  508. serializer_class = serializers.PowerFeedSerializer
  509. filterset_class = filters.PowerFeedFilterSet
  510. #
  511. # Miscellaneous
  512. #
  513. class ConnectedDeviceViewSet(ViewSet):
  514. """
  515. This endpoint allows a user to determine what device (if any) is connected to a given peer device and peer
  516. interface. This is useful in a situation where a device boots with no configuration, but can detect its neighbors
  517. via a protocol such as LLDP. Two query parameters must be included in the request:
  518. * `peer_device`: The name of the peer device
  519. * `peer_interface`: The name of the peer interface
  520. """
  521. permission_classes = [IsAuthenticatedOrLoginNotRequired]
  522. _device_param = Parameter(
  523. name='peer_device',
  524. in_='query',
  525. description='The name of the peer device',
  526. required=True,
  527. type=openapi.TYPE_STRING
  528. )
  529. _interface_param = Parameter(
  530. name='peer_interface',
  531. in_='query',
  532. description='The name of the peer interface',
  533. required=True,
  534. type=openapi.TYPE_STRING
  535. )
  536. def get_view_name(self):
  537. return "Connected Device Locator"
  538. @swagger_auto_schema(
  539. manual_parameters=[_device_param, _interface_param],
  540. responses={'200': serializers.DeviceSerializer}
  541. )
  542. def list(self, request):
  543. peer_device_name = request.query_params.get(self._device_param.name)
  544. peer_interface_name = request.query_params.get(self._interface_param.name)
  545. if not peer_device_name or not peer_interface_name:
  546. raise MissingFilterException(detail='Request must include "peer_device" and "peer_interface" filters.')
  547. # Determine local interface from peer interface's connection
  548. peer_interface = get_object_or_404(Interface, device__name=peer_device_name, name=peer_interface_name)
  549. local_interface = peer_interface._connected_interface
  550. if local_interface is None:
  551. return Response()
  552. return Response(serializers.DeviceSerializer(local_interface.device, context={'request': request}).data)