views.py 13 KB

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