views.py 24 KB

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