views.py 20 KB

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