views.py 15 KB

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