views.py 16 KB

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