views.py 23 KB

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