views.py 16 KB

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