views.py 24 KB

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