views.py 15 KB

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