views.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497
  1. from rest_framework import generics
  2. from rest_framework.permissions import DjangoModelPermissionsOrAnonReadOnly
  3. from rest_framework.response import Response
  4. from rest_framework.settings import api_settings
  5. from rest_framework.views import APIView
  6. from django.conf import settings
  7. from django.contrib.contenttypes.models import ContentType
  8. from django.http import Http404
  9. from django.shortcuts import get_object_or_404
  10. from dcim.models import (
  11. ConsolePort, ConsoleServerPort, Device, DeviceBay, DeviceRole, DeviceType, IFACE_FF_VIRTUAL, Interface,
  12. InterfaceConnection, Manufacturer, Module, Platform, PowerOutlet, PowerPort, Rack, RackGroup, RackRole, Site,
  13. )
  14. from dcim import filters
  15. from extras.api.views import CustomFieldModelAPIView
  16. from extras.api.renderers import BINDZoneRenderer, FlatJSONRenderer
  17. from utilities.api import ServiceUnavailable
  18. from .exceptions import MissingFilterException
  19. from . import serializers
  20. #
  21. # Sites
  22. #
  23. class SiteListView(CustomFieldModelAPIView, generics.ListAPIView):
  24. """
  25. List all sites
  26. """
  27. queryset = Site.objects.select_related('tenant').prefetch_related('custom_field_values__field')
  28. serializer_class = serializers.SiteSerializer
  29. class SiteDetailView(CustomFieldModelAPIView, generics.RetrieveAPIView):
  30. """
  31. Retrieve a single site
  32. """
  33. queryset = Site.objects.select_related('tenant').prefetch_related('custom_field_values__field')
  34. serializer_class = serializers.SiteSerializer
  35. #
  36. # Rack groups
  37. #
  38. class RackGroupListView(generics.ListAPIView):
  39. """
  40. List all rack groups
  41. """
  42. queryset = RackGroup.objects.select_related('site')
  43. serializer_class = serializers.RackGroupSerializer
  44. filter_class = filters.RackGroupFilter
  45. class RackGroupDetailView(generics.RetrieveAPIView):
  46. """
  47. Retrieve a single rack group
  48. """
  49. queryset = RackGroup.objects.select_related('site')
  50. serializer_class = serializers.RackGroupSerializer
  51. #
  52. # Rack roles
  53. #
  54. class RackRoleListView(generics.ListAPIView):
  55. """
  56. List all rack roles
  57. """
  58. queryset = RackRole.objects.all()
  59. serializer_class = serializers.RackRoleSerializer
  60. class RackRoleDetailView(generics.RetrieveAPIView):
  61. """
  62. Retrieve a single rack role
  63. """
  64. queryset = RackRole.objects.all()
  65. serializer_class = serializers.RackRoleSerializer
  66. #
  67. # Racks
  68. #
  69. class RackListView(CustomFieldModelAPIView, generics.ListAPIView):
  70. """
  71. List racks (filterable)
  72. """
  73. queryset = Rack.objects.select_related('site', 'group__site', 'tenant')\
  74. .prefetch_related('custom_field_values__field')
  75. serializer_class = serializers.RackSerializer
  76. filter_class = filters.RackFilter
  77. class RackDetailView(CustomFieldModelAPIView, generics.RetrieveAPIView):
  78. """
  79. Retrieve a single rack
  80. """
  81. queryset = Rack.objects.select_related('site', 'group__site', 'tenant')\
  82. .prefetch_related('custom_field_values__field')
  83. serializer_class = serializers.RackDetailSerializer
  84. #
  85. # Rack units
  86. #
  87. class RackUnitListView(APIView):
  88. """
  89. List rack units (by rack)
  90. """
  91. def get(self, request, pk):
  92. rack = get_object_or_404(Rack, pk=pk)
  93. face = request.GET.get('face', 0)
  94. try:
  95. exclude = int(request.GET.get('exclude', None))
  96. except ValueError:
  97. exclude = None
  98. elevation = rack.get_rack_units(face, exclude)
  99. # Serialize Devices within the rack elevation
  100. for u in elevation:
  101. if u['device']:
  102. u['device'] = serializers.DeviceNestedSerializer(instance=u['device']).data
  103. return Response(elevation)
  104. #
  105. # Manufacturers
  106. #
  107. class ManufacturerListView(generics.ListAPIView):
  108. """
  109. List all hardware manufacturers
  110. """
  111. queryset = Manufacturer.objects.all()
  112. serializer_class = serializers.ManufacturerSerializer
  113. class ManufacturerDetailView(generics.RetrieveAPIView):
  114. """
  115. Retrieve a single hardware manufacturers
  116. """
  117. queryset = Manufacturer.objects.all()
  118. serializer_class = serializers.ManufacturerSerializer
  119. #
  120. # Device Types
  121. #
  122. class DeviceTypeListView(CustomFieldModelAPIView, generics.ListAPIView):
  123. """
  124. List device types (filterable)
  125. """
  126. queryset = DeviceType.objects.select_related('manufacturer').prefetch_related('custom_field_values__field')
  127. serializer_class = serializers.DeviceTypeSerializer
  128. filter_class = filters.DeviceTypeFilter
  129. class DeviceTypeDetailView(CustomFieldModelAPIView, generics.RetrieveAPIView):
  130. """
  131. Retrieve a single device type
  132. """
  133. queryset = DeviceType.objects.select_related('manufacturer').prefetch_related('custom_field_values__field')
  134. serializer_class = serializers.DeviceTypeDetailSerializer
  135. #
  136. # Device roles
  137. #
  138. class DeviceRoleListView(generics.ListAPIView):
  139. """
  140. List all device roles
  141. """
  142. queryset = DeviceRole.objects.all()
  143. serializer_class = serializers.DeviceRoleSerializer
  144. class DeviceRoleDetailView(generics.RetrieveAPIView):
  145. """
  146. Retrieve a single device role
  147. """
  148. queryset = DeviceRole.objects.all()
  149. serializer_class = serializers.DeviceRoleSerializer
  150. #
  151. # Platforms
  152. #
  153. class PlatformListView(generics.ListAPIView):
  154. """
  155. List all platforms
  156. """
  157. queryset = Platform.objects.all()
  158. serializer_class = serializers.PlatformSerializer
  159. class PlatformDetailView(generics.RetrieveAPIView):
  160. """
  161. Retrieve a single platform
  162. """
  163. queryset = Platform.objects.all()
  164. serializer_class = serializers.PlatformSerializer
  165. #
  166. # Devices
  167. #
  168. class DeviceListView(CustomFieldModelAPIView, generics.ListAPIView):
  169. """
  170. List devices (filterable)
  171. """
  172. queryset = Device.objects.select_related('device_type__manufacturer', 'device_role', 'tenant', 'platform',
  173. 'rack__site', 'parent_bay').prefetch_related('primary_ip4__nat_outside',
  174. 'primary_ip6__nat_outside',
  175. 'custom_field_values__field')
  176. serializer_class = serializers.DeviceSerializer
  177. filter_class = filters.DeviceFilter
  178. renderer_classes = api_settings.DEFAULT_RENDERER_CLASSES + [BINDZoneRenderer, FlatJSONRenderer]
  179. class DeviceDetailView(CustomFieldModelAPIView, generics.RetrieveAPIView):
  180. """
  181. Retrieve a single device
  182. """
  183. queryset = Device.objects.select_related('device_type__manufacturer', 'device_role', 'tenant', 'platform',
  184. 'rack__site', 'parent_bay').prefetch_related('custom_field_values__field')
  185. serializer_class = serializers.DeviceSerializer
  186. #
  187. # Console ports
  188. #
  189. class ConsolePortListView(generics.ListAPIView):
  190. """
  191. List console ports (by device)
  192. """
  193. serializer_class = serializers.ConsolePortSerializer
  194. def get_queryset(self):
  195. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  196. return ConsolePort.objects.filter(device=device).select_related('cs_port')
  197. class ConsolePortView(generics.RetrieveUpdateDestroyAPIView):
  198. permission_classes = [DjangoModelPermissionsOrAnonReadOnly]
  199. serializer_class = serializers.ConsolePortSerializer
  200. queryset = ConsolePort.objects.all()
  201. #
  202. # Console server ports
  203. #
  204. class ConsoleServerPortListView(generics.ListAPIView):
  205. """
  206. List console server ports (by device)
  207. """
  208. serializer_class = serializers.ConsoleServerPortSerializer
  209. def get_queryset(self):
  210. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  211. return ConsoleServerPort.objects.filter(device=device).select_related('connected_console')
  212. #
  213. # Power ports
  214. #
  215. class PowerPortListView(generics.ListAPIView):
  216. """
  217. List power ports (by device)
  218. """
  219. serializer_class = serializers.PowerPortSerializer
  220. def get_queryset(self):
  221. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  222. return PowerPort.objects.filter(device=device).select_related('power_outlet')
  223. class PowerPortView(generics.RetrieveUpdateDestroyAPIView):
  224. permission_classes = [DjangoModelPermissionsOrAnonReadOnly]
  225. serializer_class = serializers.PowerPortSerializer
  226. queryset = PowerPort.objects.all()
  227. #
  228. # Power outlets
  229. #
  230. class PowerOutletListView(generics.ListAPIView):
  231. """
  232. List power outlets (by device)
  233. """
  234. serializer_class = serializers.PowerOutletSerializer
  235. def get_queryset(self):
  236. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  237. return PowerOutlet.objects.filter(device=device).select_related('connected_port')
  238. #
  239. # Interfaces
  240. #
  241. class InterfaceListView(generics.ListAPIView):
  242. """
  243. List interfaces (by device)
  244. """
  245. serializer_class = serializers.InterfaceSerializer
  246. filter_class = filters.InterfaceFilter
  247. def get_queryset(self):
  248. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  249. queryset = Interface.objects.filter(device=device).select_related('connected_as_a', 'connected_as_b')
  250. # Filter by type (physical or virtual)
  251. iface_type = self.request.query_params.get('type')
  252. if iface_type == 'physical':
  253. queryset = queryset.exclude(form_factor=IFACE_FF_VIRTUAL)
  254. elif iface_type == 'virtual':
  255. queryset = queryset.filter(form_factor=IFACE_FF_VIRTUAL)
  256. elif iface_type is not None:
  257. queryset = queryset.empty()
  258. return queryset
  259. class InterfaceDetailView(generics.RetrieveAPIView):
  260. """
  261. Retrieve a single interface
  262. """
  263. queryset = Interface.objects.select_related('device')
  264. serializer_class = serializers.InterfaceDetailSerializer
  265. class InterfaceConnectionView(generics.RetrieveUpdateDestroyAPIView):
  266. permission_classes = [DjangoModelPermissionsOrAnonReadOnly]
  267. serializer_class = serializers.InterfaceConnectionSerializer
  268. queryset = InterfaceConnection.objects.all()
  269. class InterfaceConnectionListView(generics.ListAPIView):
  270. """
  271. Retrieve a list of all interface connections
  272. """
  273. serializer_class = serializers.InterfaceConnectionSerializer
  274. queryset = InterfaceConnection.objects.all()
  275. #
  276. # Device bays
  277. #
  278. class DeviceBayListView(generics.ListAPIView):
  279. """
  280. List device bays (by device)
  281. """
  282. serializer_class = serializers.DeviceBayNestedSerializer
  283. def get_queryset(self):
  284. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  285. return DeviceBay.objects.filter(device=device).select_related('installed_device')
  286. #
  287. # Modules
  288. #
  289. class ModuleListView(generics.ListAPIView):
  290. """
  291. List device modules (by device)
  292. """
  293. serializer_class = serializers.ModuleSerializer
  294. def get_queryset(self):
  295. device = get_object_or_404(Device, pk=self.kwargs['pk'])
  296. return Module.objects.filter(device=device).select_related('device', 'manufacturer')
  297. #
  298. # Live queries
  299. #
  300. class LLDPNeighborsView(APIView):
  301. """
  302. Retrieve live LLDP neighbors of a device
  303. """
  304. def get(self, request, pk):
  305. device = get_object_or_404(Device, pk=pk)
  306. if not device.primary_ip:
  307. raise ServiceUnavailable(detail="No IP configured for this device.")
  308. RPC = device.get_rpc_client()
  309. if not RPC:
  310. raise ServiceUnavailable(detail="No RPC client available for this platform ({}).".format(device.platform))
  311. # Connect to device and retrieve inventory info
  312. try:
  313. with RPC(device, username=settings.NETBOX_USERNAME, password=settings.NETBOX_PASSWORD) as rpc_client:
  314. lldp_neighbors = rpc_client.get_lldp_neighbors()
  315. except:
  316. raise ServiceUnavailable(detail="Error connecting to the remote device.")
  317. return Response(lldp_neighbors)
  318. #
  319. # Miscellaneous
  320. #
  321. class RelatedConnectionsView(APIView):
  322. """
  323. Retrieve all connections related to a given console/power/interface connection
  324. """
  325. def __init__(self):
  326. super(RelatedConnectionsView, self).__init__()
  327. # Custom fields
  328. self.content_type = ContentType.objects.get_for_model(Device)
  329. self.custom_fields = self.content_type.custom_fields.prefetch_related('choices')
  330. def get(self, request):
  331. peer_device = request.GET.get('peer-device')
  332. peer_interface = request.GET.get('peer-interface')
  333. # Search by interface
  334. if peer_device and peer_interface:
  335. # Determine local interface from peer interface's connection
  336. try:
  337. peer_iface = Interface.objects.get(device__name=peer_device, name=peer_interface)
  338. except Interface.DoesNotExist:
  339. raise Http404()
  340. local_iface = peer_iface.connected_interface
  341. if local_iface:
  342. device = local_iface.device
  343. else:
  344. return Response()
  345. else:
  346. raise MissingFilterException(detail='Must specify search parameters "peer-device" and "peer-interface".')
  347. # Initialize response skeleton
  348. response = {
  349. 'device': serializers.DeviceSerializer(device, context={'view': self}).data,
  350. 'console-ports': [],
  351. 'power-ports': [],
  352. 'interfaces': [],
  353. }
  354. # Console connections
  355. console_ports = ConsolePort.objects.filter(device=device).select_related('cs_port__device')
  356. for cp in console_ports:
  357. data = serializers.ConsolePortSerializer(instance=cp).data
  358. del(data['device'])
  359. response['console-ports'].append(data)
  360. # Power connections
  361. power_ports = PowerPort.objects.filter(device=device).select_related('power_outlet__device')
  362. for pp in power_ports:
  363. data = serializers.PowerPortSerializer(instance=pp).data
  364. del(data['device'])
  365. response['power-ports'].append(data)
  366. # Interface connections
  367. interfaces = Interface.objects.filter(device=device).select_related('connected_as_a', 'connected_as_b',
  368. 'circuit_termination')
  369. for iface in interfaces:
  370. data = serializers.InterfaceDetailSerializer(instance=iface).data
  371. del(data['device'])
  372. response['interfaces'].append(data)
  373. return Response(response)