views.py 22 KB

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