views.py 15 KB

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