views.py 12 KB

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