views.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451
  1. from __future__ import unicode_literals
  2. from collections import OrderedDict
  3. from django.conf import settings
  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. ConsolePort, ConsolePortTemplate, ConsoleServerPort, ConsoleServerPortTemplate, Device, DeviceBay,
  16. DeviceBayTemplate, DeviceRole, DeviceType, Interface, InterfaceConnection, InterfaceTemplate, Manufacturer,
  17. InventoryItem, Platform, PowerOutlet, PowerOutletTemplate, PowerPort, PowerPortTemplate, Rack, RackGroup,
  18. RackReservation, RackRole, Region, Site, VirtualChassis,
  19. )
  20. from extras.api.serializers import RenderedGraphSerializer
  21. from extras.api.views import CustomFieldModelViewSet
  22. from extras.models import Graph, GRAPH_TYPE_INTERFACE, GRAPH_TYPE_SITE
  23. from utilities.api import IsAuthenticatedOrLoginNotRequired, FieldChoicesViewSet, ModelViewSet, ServiceUnavailable
  24. from . import serializers
  25. from .exceptions import MissingFilterException
  26. #
  27. # Field choices
  28. #
  29. class DCIMFieldChoicesViewSet(FieldChoicesViewSet):
  30. fields = (
  31. (Device, ['face', 'status']),
  32. (ConsolePort, ['connection_status']),
  33. (Interface, ['form_factor', 'mode']),
  34. (InterfaceConnection, ['connection_status']),
  35. (InterfaceTemplate, ['form_factor']),
  36. (PowerPort, ['connection_status']),
  37. (Rack, ['type', 'width']),
  38. (Site, ['status']),
  39. )
  40. #
  41. # Regions
  42. #
  43. class RegionViewSet(ModelViewSet):
  44. queryset = Region.objects.all()
  45. serializer_class = serializers.RegionSerializer
  46. filter_class = filters.RegionFilter
  47. #
  48. # Sites
  49. #
  50. class SiteViewSet(CustomFieldModelViewSet):
  51. queryset = Site.objects.select_related('region', 'tenant').prefetch_related('tags')
  52. serializer_class = serializers.SiteSerializer
  53. filter_class = filters.SiteFilter
  54. @action(detail=True)
  55. def graphs(self, request, pk=None):
  56. """
  57. A convenience method for rendering graphs for a particular site.
  58. """
  59. site = get_object_or_404(Site, pk=pk)
  60. queryset = Graph.objects.filter(type=GRAPH_TYPE_SITE)
  61. serializer = RenderedGraphSerializer(queryset, many=True, context={'graphed_object': site})
  62. return Response(serializer.data)
  63. #
  64. # Rack groups
  65. #
  66. class RackGroupViewSet(ModelViewSet):
  67. queryset = RackGroup.objects.select_related('site')
  68. serializer_class = serializers.RackGroupSerializer
  69. filter_class = filters.RackGroupFilter
  70. #
  71. # Rack roles
  72. #
  73. class RackRoleViewSet(ModelViewSet):
  74. queryset = RackRole.objects.all()
  75. serializer_class = serializers.RackRoleSerializer
  76. filter_class = filters.RackRoleFilter
  77. #
  78. # Racks
  79. #
  80. class RackViewSet(CustomFieldModelViewSet):
  81. queryset = Rack.objects.select_related('site', 'group__site', 'tenant').prefetch_related('tags')
  82. serializer_class = serializers.RackSerializer
  83. filter_class = filters.RackFilter
  84. @action(detail=True)
  85. def units(self, request, pk=None):
  86. """
  87. List rack units (by rack)
  88. """
  89. rack = get_object_or_404(Rack, pk=pk)
  90. face = request.GET.get('face', 0)
  91. exclude_pk = request.GET.get('exclude', None)
  92. if exclude_pk is not None:
  93. try:
  94. exclude_pk = int(exclude_pk)
  95. except ValueError:
  96. exclude_pk = None
  97. elevation = rack.get_rack_units(face, exclude_pk)
  98. page = self.paginate_queryset(elevation)
  99. if page is not None:
  100. rack_units = serializers.RackUnitSerializer(page, many=True, context={'request': request})
  101. return self.get_paginated_response(rack_units.data)
  102. #
  103. # Rack reservations
  104. #
  105. class RackReservationViewSet(ModelViewSet):
  106. queryset = RackReservation.objects.select_related('rack', 'user', 'tenant')
  107. serializer_class = serializers.RackReservationSerializer
  108. filter_class = filters.RackReservationFilter
  109. # Assign user from request
  110. def perform_create(self, serializer):
  111. serializer.save(user=self.request.user)
  112. #
  113. # Manufacturers
  114. #
  115. class ManufacturerViewSet(ModelViewSet):
  116. queryset = Manufacturer.objects.all()
  117. serializer_class = serializers.ManufacturerSerializer
  118. filter_class = filters.ManufacturerFilter
  119. #
  120. # Device types
  121. #
  122. class DeviceTypeViewSet(CustomFieldModelViewSet):
  123. queryset = DeviceType.objects.select_related('manufacturer').prefetch_related('tags')
  124. serializer_class = serializers.DeviceTypeSerializer
  125. filter_class = filters.DeviceTypeFilter
  126. #
  127. # Device type components
  128. #
  129. class ConsolePortTemplateViewSet(ModelViewSet):
  130. queryset = ConsolePortTemplate.objects.select_related('device_type__manufacturer')
  131. serializer_class = serializers.ConsolePortTemplateSerializer
  132. filter_class = filters.ConsolePortTemplateFilter
  133. class ConsoleServerPortTemplateViewSet(ModelViewSet):
  134. queryset = ConsoleServerPortTemplate.objects.select_related('device_type__manufacturer')
  135. serializer_class = serializers.ConsoleServerPortTemplateSerializer
  136. filter_class = filters.ConsoleServerPortTemplateFilter
  137. class PowerPortTemplateViewSet(ModelViewSet):
  138. queryset = PowerPortTemplate.objects.select_related('device_type__manufacturer')
  139. serializer_class = serializers.PowerPortTemplateSerializer
  140. filter_class = filters.PowerPortTemplateFilter
  141. class PowerOutletTemplateViewSet(ModelViewSet):
  142. queryset = PowerOutletTemplate.objects.select_related('device_type__manufacturer')
  143. serializer_class = serializers.PowerOutletTemplateSerializer
  144. filter_class = filters.PowerOutletTemplateFilter
  145. class InterfaceTemplateViewSet(ModelViewSet):
  146. queryset = InterfaceTemplate.objects.select_related('device_type__manufacturer')
  147. serializer_class = serializers.InterfaceTemplateSerializer
  148. filter_class = filters.InterfaceTemplateFilter
  149. class DeviceBayTemplateViewSet(ModelViewSet):
  150. queryset = DeviceBayTemplate.objects.select_related('device_type__manufacturer')
  151. serializer_class = serializers.DeviceBayTemplateSerializer
  152. filter_class = filters.DeviceBayTemplateFilter
  153. #
  154. # Device roles
  155. #
  156. class DeviceRoleViewSet(ModelViewSet):
  157. queryset = DeviceRole.objects.all()
  158. serializer_class = serializers.DeviceRoleSerializer
  159. filter_class = filters.DeviceRoleFilter
  160. #
  161. # Platforms
  162. #
  163. class PlatformViewSet(ModelViewSet):
  164. queryset = Platform.objects.all()
  165. serializer_class = serializers.PlatformSerializer
  166. filter_class = filters.PlatformFilter
  167. #
  168. # Devices
  169. #
  170. class DeviceViewSet(CustomFieldModelViewSet):
  171. queryset = Device.objects.select_related(
  172. 'device_type__manufacturer', 'device_role', 'tenant', 'platform', 'site', 'rack', 'parent_bay',
  173. 'virtual_chassis__master',
  174. ).prefetch_related(
  175. 'primary_ip4__nat_outside', 'primary_ip6__nat_outside', 'tags',
  176. )
  177. filter_class = filters.DeviceFilter
  178. def get_serializer_class(self):
  179. """
  180. Include rendered config context when retrieving a single Device.
  181. """
  182. if self.action == 'retrieve':
  183. return serializers.DeviceWithConfigContextSerializer
  184. request = self.get_serializer_context()['request']
  185. if request.query_params.get('brief', False):
  186. return serializers.NestedDeviceSerializer
  187. return serializers.DeviceSerializer
  188. @action(detail=True, url_path='napalm')
  189. def napalm(self, request, pk):
  190. """
  191. Execute a NAPALM method on a Device
  192. """
  193. device = get_object_or_404(Device, pk=pk)
  194. if not device.primary_ip:
  195. raise ServiceUnavailable("This device does not have a primary IP address configured.")
  196. if device.platform is None:
  197. raise ServiceUnavailable("No platform is configured for this device.")
  198. if not device.platform.napalm_driver:
  199. raise ServiceUnavailable("No NAPALM driver is configured for this device's platform ().".format(
  200. device.platform
  201. ))
  202. # Check that NAPALM is installed
  203. try:
  204. import napalm
  205. from napalm.base.exceptions import ModuleImportError
  206. except ImportError:
  207. raise ServiceUnavailable("NAPALM is not installed. Please see the documentation for instructions.")
  208. # Validate the configured driver
  209. try:
  210. driver = napalm.get_network_driver(device.platform.napalm_driver)
  211. except ModuleImportError:
  212. raise ServiceUnavailable("NAPALM driver for platform {} not found: {}.".format(
  213. device.platform, device.platform.napalm_driver
  214. ))
  215. # Verify user permission
  216. if not request.user.has_perm('dcim.napalm_read'):
  217. return HttpResponseForbidden()
  218. # Connect to the device
  219. napalm_methods = request.GET.getlist('method')
  220. response = OrderedDict([(m, None) for m in napalm_methods])
  221. ip_address = str(device.primary_ip.address.ip)
  222. optional_args = settings.NAPALM_ARGS.copy()
  223. if device.platform.napalm_args is not None:
  224. optional_args.update(device.platform.napalm_args)
  225. d = driver(
  226. hostname=ip_address,
  227. username=settings.NAPALM_USERNAME,
  228. password=settings.NAPALM_PASSWORD,
  229. timeout=settings.NAPALM_TIMEOUT,
  230. optional_args=optional_args
  231. )
  232. try:
  233. d.open()
  234. except Exception as e:
  235. raise ServiceUnavailable("Error connecting to the device at {}: {}".format(ip_address, e))
  236. # Validate and execute each specified NAPALM method
  237. for method in napalm_methods:
  238. if not hasattr(driver, method):
  239. response[method] = {'error': 'Unknown NAPALM method'}
  240. continue
  241. if not method.startswith('get_'):
  242. response[method] = {'error': 'Only get_* NAPALM methods are supported'}
  243. continue
  244. try:
  245. response[method] = getattr(d, method)()
  246. except NotImplementedError:
  247. response[method] = {'error': 'Method {} not implemented for NAPALM driver {}'.format(method, driver)}
  248. except Exception as e:
  249. response[method] = {'error': 'Method {} failed: {}'.format(method, e)}
  250. d.close()
  251. return Response(response)
  252. #
  253. # Device components
  254. #
  255. class ConsolePortViewSet(ModelViewSet):
  256. queryset = ConsolePort.objects.select_related('device', 'cs_port__device').prefetch_related('tags')
  257. serializer_class = serializers.ConsolePortSerializer
  258. filter_class = filters.ConsolePortFilter
  259. class ConsoleServerPortViewSet(ModelViewSet):
  260. queryset = ConsoleServerPort.objects.select_related('device', 'connected_console__device').prefetch_related('tags')
  261. serializer_class = serializers.ConsoleServerPortSerializer
  262. filter_class = filters.ConsoleServerPortFilter
  263. class PowerPortViewSet(ModelViewSet):
  264. queryset = PowerPort.objects.select_related('device', 'power_outlet__device').prefetch_related('tags')
  265. serializer_class = serializers.PowerPortSerializer
  266. filter_class = filters.PowerPortFilter
  267. class PowerOutletViewSet(ModelViewSet):
  268. queryset = PowerOutlet.objects.select_related('device', 'connected_port__device').prefetch_related('tags')
  269. serializer_class = serializers.PowerOutletSerializer
  270. filter_class = filters.PowerOutletFilter
  271. class InterfaceViewSet(ModelViewSet):
  272. queryset = Interface.objects.select_related('device').prefetch_related('tags')
  273. serializer_class = serializers.InterfaceSerializer
  274. filter_class = filters.InterfaceFilter
  275. @action(detail=True)
  276. def graphs(self, request, pk=None):
  277. """
  278. A convenience method for rendering graphs for a particular interface.
  279. """
  280. interface = get_object_or_404(Interface, pk=pk)
  281. queryset = Graph.objects.filter(type=GRAPH_TYPE_INTERFACE)
  282. serializer = RenderedGraphSerializer(queryset, many=True, context={'graphed_object': interface})
  283. return Response(serializer.data)
  284. class DeviceBayViewSet(ModelViewSet):
  285. queryset = DeviceBay.objects.select_related('installed_device').prefetch_related('tags')
  286. serializer_class = serializers.DeviceBaySerializer
  287. filter_class = filters.DeviceBayFilter
  288. class InventoryItemViewSet(ModelViewSet):
  289. queryset = InventoryItem.objects.select_related('device', 'manufacturer').prefetch_related('tags')
  290. serializer_class = serializers.InventoryItemSerializer
  291. filter_class = filters.InventoryItemFilter
  292. #
  293. # Connections
  294. #
  295. class ConsoleConnectionViewSet(ListModelMixin, GenericViewSet):
  296. queryset = ConsolePort.objects.select_related('device', 'cs_port__device').filter(cs_port__isnull=False)
  297. serializer_class = serializers.ConsolePortSerializer
  298. filter_class = filters.ConsoleConnectionFilter
  299. class PowerConnectionViewSet(ListModelMixin, GenericViewSet):
  300. queryset = PowerPort.objects.select_related('device', 'power_outlet__device').filter(power_outlet__isnull=False)
  301. serializer_class = serializers.PowerPortSerializer
  302. filter_class = filters.PowerConnectionFilter
  303. class InterfaceConnectionViewSet(ModelViewSet):
  304. queryset = InterfaceConnection.objects.select_related('interface_a__device', 'interface_b__device')
  305. serializer_class = serializers.InterfaceConnectionSerializer
  306. filter_class = filters.InterfaceConnectionFilter
  307. #
  308. # Virtual chassis
  309. #
  310. class VirtualChassisViewSet(ModelViewSet):
  311. queryset = VirtualChassis.objects.prefetch_related('tags')
  312. serializer_class = serializers.VirtualChassisSerializer
  313. #
  314. # Miscellaneous
  315. #
  316. class ConnectedDeviceViewSet(ViewSet):
  317. """
  318. This endpoint allows a user to determine what device (if any) is connected to a given peer device and peer
  319. interface. This is useful in a situation where a device boots with no configuration, but can detect its neighbors
  320. via a protocol such as LLDP. Two query parameters must be included in the request:
  321. * `peer_device`: The name of the peer device
  322. * `peer_interface`: The name of the peer interface
  323. """
  324. permission_classes = [IsAuthenticatedOrLoginNotRequired]
  325. _device_param = Parameter('peer_device', 'query',
  326. description='The name of the peer device', required=True, type=openapi.TYPE_STRING)
  327. _interface_param = Parameter('peer_interface', 'query',
  328. description='The name of the peer interface', required=True, type=openapi.TYPE_STRING)
  329. def get_view_name(self):
  330. return "Connected Device Locator"
  331. @swagger_auto_schema(
  332. manual_parameters=[_device_param, _interface_param], responses={'200': serializers.DeviceSerializer})
  333. def list(self, request):
  334. peer_device_name = request.query_params.get(self._device_param.name)
  335. if not peer_device_name:
  336. # TODO: remove this after 2.4 as the switch to using underscores is a breaking change
  337. peer_device_name = request.query_params.get('peer-device')
  338. peer_interface_name = request.query_params.get(self._interface_param.name)
  339. if not peer_interface_name:
  340. # TODO: remove this after 2.4 as the switch to using underscores is a breaking change
  341. peer_interface_name = request.query_params.get('peer-interface')
  342. if not peer_device_name or not peer_interface_name:
  343. raise MissingFilterException(detail='Request must include "peer_device" and "peer_interface" filters.')
  344. # Determine local interface from peer interface's connection
  345. peer_interface = get_object_or_404(Interface, device__name=peer_device_name, name=peer_interface_name)
  346. local_interface = peer_interface.connected_interface
  347. if local_interface is None:
  348. return Response()
  349. return Response(serializers.DeviceSerializer(local_interface.device, context={'request': request}).data)