views.py 15 KB

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