views.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387
  1. from rest_framework.mixins import (
  2. CreateModelMixin, DestroyModelMixin, ListModelMixin, RetrieveModelMixin, UpdateModelMixin,
  3. )
  4. from rest_framework.response import Response
  5. from rest_framework.settings import api_settings
  6. from rest_framework.views import APIView
  7. from rest_framework.viewsets import GenericViewSet, ModelViewSet
  8. from django.conf import settings
  9. from django.contrib.contenttypes.models import ContentType
  10. from django.http import Http404
  11. from django.shortcuts import get_object_or_404
  12. from dcim.models import (
  13. ConsolePort, ConsoleServerPort, Device, DeviceBay, DeviceRole, DeviceType, Interface, InterfaceConnection,
  14. Manufacturer, Module, Platform, PowerOutlet, PowerPort, Rack, RackGroup, RackRole, Site,
  15. )
  16. from dcim import filters
  17. from extras.api.views import CustomFieldModelViewSet
  18. from extras.api.renderers import BINDZoneRenderer, FlatJSONRenderer
  19. from utilities.api import ServiceUnavailable, WritableSerializerMixin
  20. from .exceptions import MissingFilterException
  21. from . import serializers
  22. #
  23. # Sites
  24. #
  25. class SiteViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
  26. queryset = Site.objects.select_related('tenant')
  27. serializer_class = serializers.SiteSerializer
  28. #
  29. # Rack groups
  30. #
  31. class RackGroupViewSet(WritableSerializerMixin, ModelViewSet):
  32. queryset = RackGroup.objects.select_related('site')
  33. serializer_class = serializers.RackGroupSerializer
  34. filter_class = filters.RackGroupFilter
  35. #
  36. # Rack roles
  37. #
  38. class RackRoleViewSet(ModelViewSet):
  39. queryset = RackRole.objects.all()
  40. serializer_class = serializers.RackRoleSerializer
  41. #
  42. # Racks
  43. #
  44. class RackViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
  45. queryset = Rack.objects.select_related('site', 'group__site', 'tenant')
  46. serializer_class = serializers.RackSerializer
  47. filter_class = filters.RackFilter
  48. class RackUnitListView(APIView):
  49. """
  50. List rack units (by rack)
  51. """
  52. def get(self, request, pk):
  53. rack = get_object_or_404(Rack, pk=pk)
  54. face = request.GET.get('face', 0)
  55. exclude_pk = request.GET.get('exclude', None)
  56. if exclude_pk is not None:
  57. try:
  58. exclude_pk = int(exclude_pk)
  59. except ValueError:
  60. exclude_pk = None
  61. elevation = rack.get_rack_units(face, exclude_pk)
  62. # Serialize Devices within the rack elevation
  63. for u in elevation:
  64. if u['device']:
  65. u['device'] = serializers.NestedDeviceSerializer(
  66. instance=u['device'],
  67. context={'request': request},
  68. ).data
  69. return Response(elevation)
  70. #
  71. # Manufacturers
  72. #
  73. class ManufacturerViewSet(ModelViewSet):
  74. queryset = Manufacturer.objects.all()
  75. serializer_class = serializers.ManufacturerSerializer
  76. #
  77. # Device Types
  78. #
  79. class DeviceTypeViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
  80. queryset = DeviceType.objects.select_related('manufacturer')
  81. serializer_class = serializers.DeviceTypeSerializer
  82. #
  83. # Device Roles
  84. #
  85. class DeviceRoleViewSet(ModelViewSet):
  86. queryset = DeviceRole.objects.all()
  87. serializer_class = serializers.DeviceRoleSerializer
  88. #
  89. # Platforms
  90. #
  91. class PlatformViewSet(ModelViewSet):
  92. queryset = Platform.objects.all()
  93. serializer_class = serializers.PlatformSerializer
  94. #
  95. # Devices
  96. #
  97. class DeviceViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
  98. queryset = Device.objects.select_related(
  99. 'device_type__manufacturer', 'device_role', 'tenant', 'platform', 'rack__site', 'parent_bay',
  100. ).prefetch_related(
  101. 'primary_ip4__nat_outside', 'primary_ip6__nat_outside',
  102. )
  103. serializer_class = serializers.DeviceSerializer
  104. filter_class = filters.DeviceFilter
  105. renderer_classes = api_settings.DEFAULT_RENDERER_CLASSES + [BINDZoneRenderer, FlatJSONRenderer]
  106. #
  107. # Console Ports
  108. #
  109. class ConsolePortViewSet(RetrieveModelMixin, UpdateModelMixin, DestroyModelMixin, WritableSerializerMixin,
  110. GenericViewSet):
  111. queryset = ConsolePort.objects.select_related('cs_port')
  112. serializer_class = serializers.ConsolePortSerializer
  113. class ChildConsolePortViewSet(CreateModelMixin, ListModelMixin, WritableSerializerMixin, GenericViewSet):
  114. serializer_class = serializers.ChildConsoleServerPortSerializer
  115. def get_queryset(self):
  116. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  117. return ConsolePort.objects.filter(device=device).select_related('cs_port')
  118. #
  119. # Console Server Ports
  120. #
  121. class ConsoleServerPortViewSet(RetrieveModelMixin, UpdateModelMixin, DestroyModelMixin, WritableSerializerMixin,
  122. GenericViewSet):
  123. queryset = ConsoleServerPort.objects.select_related('connected_console')
  124. serializer_class = serializers.ConsoleServerPortSerializer
  125. class ChildConsoleServerPortViewSet(CreateModelMixin, ListModelMixin, WritableSerializerMixin, GenericViewSet):
  126. serializer_class = serializers.ChildConsoleServerPortSerializer
  127. def get_queryset(self):
  128. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  129. return ConsoleServerPort.objects.filter(device=device).select_related('connected_console')
  130. #
  131. # Power Ports
  132. #
  133. class PowerPortViewSet(RetrieveModelMixin, UpdateModelMixin, DestroyModelMixin, WritableSerializerMixin,
  134. GenericViewSet):
  135. queryset = PowerPort.objects.select_related('power_outlet')
  136. serializer_class = serializers.PowerPortSerializer
  137. class NestedPowerPortViewSet(CreateModelMixin, ListModelMixin, WritableSerializerMixin, GenericViewSet):
  138. serializer_class = serializers.ChildPowerPortSerializer
  139. def get_queryset(self):
  140. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  141. return PowerPort.objects.filter(device=device).select_related('power_outlet')
  142. #
  143. # Power Outlets
  144. #
  145. class PowerOutletViewSet(RetrieveModelMixin, UpdateModelMixin, DestroyModelMixin, WritableSerializerMixin,
  146. GenericViewSet):
  147. queryset = PowerOutlet.objects.select_related('connected_port')
  148. serializer_class = serializers.PowerOutletSerializer
  149. class NestedPowerOutletViewSet(CreateModelMixin, ListModelMixin, WritableSerializerMixin, GenericViewSet):
  150. serializer_class = serializers.ChildPowerOutletSerializer
  151. def get_queryset(self):
  152. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  153. return PowerOutlet.objects.filter(device=device).select_related('connected_port')
  154. #
  155. # Interfaces
  156. #
  157. class InterfaceViewSet(RetrieveModelMixin, UpdateModelMixin, DestroyModelMixin, WritableSerializerMixin,
  158. GenericViewSet):
  159. queryset = Interface.objects.select_related('device')
  160. serializer_class = serializers.InterfaceDetailSerializer
  161. class NestedInterfaceViewSet(CreateModelMixin, ListModelMixin, WritableSerializerMixin, GenericViewSet):
  162. serializer_class = serializers.ChildInterfaceSerializer
  163. filter_class = filters.InterfaceFilter
  164. def get_queryset(self):
  165. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  166. return Interface.objects.order_naturally(device.device_type.interface_ordering).filter(device=device)\
  167. .select_related('connected_as_a', 'connected_as_b', 'circuit_termination')
  168. #
  169. # Device bays
  170. #
  171. class DeviceBayViewSet(RetrieveModelMixin, UpdateModelMixin, DestroyModelMixin, WritableSerializerMixin,
  172. GenericViewSet):
  173. queryset = DeviceBay.objects.select_related('installed_device')
  174. serializer_class = serializers.DeviceBaySerializer
  175. class NestedDeviceBayViewSet(CreateModelMixin, ListModelMixin, WritableSerializerMixin, GenericViewSet):
  176. serializer_class = serializers.ChildDeviceBaySerializer
  177. def get_queryset(self):
  178. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  179. return DeviceBay.objects.filter(device=device).select_related('installed_device')
  180. #
  181. # Modules
  182. #
  183. class ModuleViewSet(RetrieveModelMixin, UpdateModelMixin, DestroyModelMixin, WritableSerializerMixin, GenericViewSet):
  184. queryset = Module.objects.select_related('device', 'manufacturer')
  185. serializer_class = serializers.ModuleSerializer
  186. class NestedModuleViewSet(CreateModelMixin, ListModelMixin, WritableSerializerMixin, GenericViewSet):
  187. serializer_class = serializers.ChildModuleSerializer
  188. def get_queryset(self):
  189. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  190. return Module.objects.filter(device=device).select_related('device', 'manufacturer')
  191. #
  192. # Interface connections
  193. #
  194. class InterfaceConnectionViewSet(ModelViewSet):
  195. queryset = InterfaceConnection.objects.all()
  196. serializer_class = serializers.InterfaceConnectionSerializer
  197. #
  198. # Live queries
  199. #
  200. class LLDPNeighborsView(APIView):
  201. """
  202. Retrieve live LLDP neighbors of a device
  203. """
  204. def get(self, request, pk):
  205. device = get_object_or_404(Device, pk=pk)
  206. if not device.primary_ip:
  207. raise ServiceUnavailable(detail="No IP configured for this device.")
  208. RPC = device.get_rpc_client()
  209. if not RPC:
  210. raise ServiceUnavailable(detail="No RPC client available for this platform ({}).".format(device.platform))
  211. # Connect to device and retrieve inventory info
  212. try:
  213. with RPC(device, username=settings.NETBOX_USERNAME, password=settings.NETBOX_PASSWORD) as rpc_client:
  214. lldp_neighbors = rpc_client.get_lldp_neighbors()
  215. except:
  216. raise ServiceUnavailable(detail="Error connecting to the remote device.")
  217. return Response(lldp_neighbors)
  218. #
  219. # Miscellaneous
  220. #
  221. class RelatedConnectionsView(APIView):
  222. """
  223. Retrieve all connections related to a given console/power/interface connection
  224. """
  225. def __init__(self):
  226. super(RelatedConnectionsView, self).__init__()
  227. # Custom fields
  228. content_type = ContentType.objects.get_for_model(Device)
  229. custom_fields = content_type.custom_fields.prefetch_related('choices')
  230. # Cache all relevant CustomFieldChoices. This saves us from having to do a lookup per select field per object.
  231. custom_field_choices = {}
  232. for field in custom_fields:
  233. for cfc in field.choices.all():
  234. custom_field_choices[cfc.id] = cfc.value
  235. self.context = {
  236. 'custom_fields': custom_fields,
  237. 'custom_field_choices': custom_field_choices,
  238. }
  239. def get(self, request):
  240. peer_device = request.GET.get('peer-device')
  241. peer_interface = request.GET.get('peer-interface')
  242. # Search by interface
  243. if peer_device and peer_interface:
  244. # Determine local interface from peer interface's connection
  245. try:
  246. peer_iface = Interface.objects.get(device__name=peer_device, name=peer_interface)
  247. except Interface.DoesNotExist:
  248. raise Http404()
  249. local_iface = peer_iface.connected_interface
  250. if local_iface:
  251. device = local_iface.device
  252. else:
  253. return Response()
  254. else:
  255. raise MissingFilterException(detail='Must specify search parameters "peer-device" and "peer-interface".')
  256. # Initialize response skeleton
  257. response = {
  258. 'device': serializers.DeviceSerializer(device, context=self.context).data,
  259. 'console-ports': [],
  260. 'power-ports': [],
  261. 'interfaces': [],
  262. }
  263. # Console connections
  264. console_ports = ConsolePort.objects.filter(device=device).select_related('cs_port__device')
  265. for cp in console_ports:
  266. data = serializers.ConsolePortSerializer(instance=cp).data
  267. del(data['device'])
  268. response['console-ports'].append(data)
  269. # Power connections
  270. power_ports = PowerPort.objects.filter(device=device).select_related('power_outlet__device')
  271. for pp in power_ports:
  272. data = serializers.PowerPortSerializer(instance=pp).data
  273. del(data['device'])
  274. response['power-ports'].append(data)
  275. # Interface connections
  276. interfaces = Interface.objects.order_naturally(device.device_type.interface_ordering).filter(device=device)\
  277. .select_related('connected_as_a', 'connected_as_b', 'circuit_termination')
  278. for iface in interfaces:
  279. data = serializers.InterfaceDetailSerializer(instance=iface).data
  280. del(data['device'])
  281. response['interfaces'].append(data)
  282. return Response(response)