views.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420
  1. from rest_framework.decorators import detail_route
  2. from rest_framework.response import Response
  3. from rest_framework.settings import api_settings
  4. from rest_framework.views import APIView
  5. from rest_framework.viewsets import ModelViewSet
  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, ConsolePortTemplate, ConsoleServerPort, ConsoleServerPortTemplate, Device, DeviceBay,
  12. DeviceBayTemplate, DeviceRole, DeviceType, Interface, InterfaceConnection, InterfaceTemplate, Manufacturer, Module,
  13. Platform, PowerOutlet, PowerOutletTemplate, PowerPort, PowerPortTemplate, Rack, RackGroup, RackReservation,
  14. RackRole, Region, Site,
  15. )
  16. from dcim import filters
  17. from extras.api.renderers import BINDZoneRenderer, FlatJSONRenderer
  18. from extras.api.serializers import GraphSerializer
  19. from extras.api.views import CustomFieldModelViewSet
  20. from extras.models import Graph, GRAPH_TYPE_INTERFACE, GRAPH_TYPE_SITE
  21. from utilities.api import ServiceUnavailable, WritableSerializerMixin
  22. from .exceptions import MissingFilterException
  23. from . import serializers
  24. #
  25. # Regions
  26. #
  27. class RegionViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
  28. queryset = Region.objects.all()
  29. serializer_class = serializers.RegionSerializer
  30. write_serializer_class = serializers.WritableRegionSerializer
  31. #
  32. # Sites
  33. #
  34. class SiteViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
  35. queryset = Site.objects.select_related('tenant')
  36. serializer_class = serializers.SiteSerializer
  37. write_serializer_class = serializers.WritableSiteSerializer
  38. @detail_route()
  39. def graphs(self, request, pk=None):
  40. site = get_object_or_404(Site, pk=pk)
  41. queryset = Graph.objects.filter(type=GRAPH_TYPE_SITE)
  42. serializer = GraphSerializer(queryset, many=True, context={'graphed_object': site})
  43. return Response(serializer.data)
  44. #
  45. # Rack groups
  46. #
  47. class RackGroupViewSet(WritableSerializerMixin, ModelViewSet):
  48. queryset = RackGroup.objects.select_related('site')
  49. serializer_class = serializers.RackGroupSerializer
  50. filter_class = filters.RackGroupFilter
  51. write_serializer_class = serializers.WritableRackGroupSerializer
  52. #
  53. # Rack roles
  54. #
  55. class RackRoleViewSet(ModelViewSet):
  56. queryset = RackRole.objects.all()
  57. serializer_class = serializers.RackRoleSerializer
  58. #
  59. # Racks
  60. #
  61. class RackViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
  62. queryset = Rack.objects.select_related('site', 'group__site', 'tenant')
  63. serializer_class = serializers.RackSerializer
  64. write_serializer_class = serializers.WritableRackSerializer
  65. filter_class = filters.RackFilter
  66. @detail_route(url_path='rack-units')
  67. def rack_units(self, request, pk=None):
  68. """
  69. List rack units (by rack)
  70. """
  71. rack = get_object_or_404(Rack, pk=pk)
  72. face = request.GET.get('face', 0)
  73. exclude_pk = request.GET.get('exclude', None)
  74. if exclude_pk is not None:
  75. try:
  76. exclude_pk = int(exclude_pk)
  77. except ValueError:
  78. exclude_pk = None
  79. elevation = rack.get_rack_units(face, exclude_pk)
  80. # Serialize Devices within the rack elevation
  81. for u in elevation:
  82. if u['device']:
  83. u['device'] = serializers.NestedDeviceSerializer(
  84. instance=u['device'],
  85. context={'request': request},
  86. ).data
  87. return Response(elevation)
  88. #
  89. # Rack reservations
  90. #
  91. class RackReservationViewSet(ModelViewSet):
  92. queryset = RackReservation.objects.all()
  93. serializer_class = serializers.RackReservationSerializer
  94. filter_class = filters.RackReservationFilter
  95. #
  96. # Manufacturers
  97. #
  98. class ManufacturerViewSet(ModelViewSet):
  99. queryset = Manufacturer.objects.all()
  100. serializer_class = serializers.ManufacturerSerializer
  101. #
  102. # Device types
  103. #
  104. class DeviceTypeViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
  105. queryset = DeviceType.objects.select_related('manufacturer')
  106. serializer_class = serializers.DeviceTypeSerializer
  107. write_serializer_class = serializers.WritableDeviceTypeSerializer
  108. #
  109. # Device type components
  110. #
  111. class ConsolePortTemplateViewSet(WritableSerializerMixin, ModelViewSet):
  112. queryset = ConsolePortTemplate.objects.select_related('device_type__manufacturer')
  113. serializer_class = serializers.ConsolePortTemplateSerializer
  114. write_serializer_class = serializers.WritableConsolePortTemplateSerializer
  115. filter_class = filters.ConsolePortTemplateFilter
  116. class ConsoleServerPortTemplateViewSet(WritableSerializerMixin, ModelViewSet):
  117. queryset = ConsoleServerPortTemplate.objects.select_related('device_type__manufacturer')
  118. serializer_class = serializers.ConsoleServerPortTemplateSerializer
  119. write_serializer_class = serializers.WritableConsoleServerPortTemplateSerializer
  120. filter_class = filters.ConsoleServerPortTemplateFilter
  121. class PowerPortTemplateViewSet(WritableSerializerMixin, ModelViewSet):
  122. queryset = PowerPortTemplate.objects.select_related('device_type__manufacturer')
  123. serializer_class = serializers.PowerPortTemplateSerializer
  124. write_serializer_class = serializers.WritablePowerPortTemplateSerializer
  125. filter_class = filters.PowerPortTemplateFilter
  126. class PowerOutletTemplateViewSet(WritableSerializerMixin, ModelViewSet):
  127. queryset = PowerOutletTemplate.objects.select_related('device_type__manufacturer')
  128. serializer_class = serializers.PowerOutletTemplateSerializer
  129. write_serializer_class = serializers.WritablePowerOutletTemplateSerializer
  130. filter_class = filters.PowerOutletTemplateFilter
  131. class InterfaceTemplateViewSet(WritableSerializerMixin, ModelViewSet):
  132. queryset = InterfaceTemplate.objects.select_related('device_type__manufacturer')
  133. serializer_class = serializers.InterfaceTemplateSerializer
  134. write_serializer_class = serializers.WritableInterfaceTemplateSerializer
  135. filter_class = filters.InterfaceTemplateFilter
  136. class DeviceBayTemplateViewSet(WritableSerializerMixin, ModelViewSet):
  137. queryset = DeviceBayTemplate.objects.select_related('device_type__manufacturer')
  138. serializer_class = serializers.DeviceBayTemplateSerializer
  139. write_serializer_class = serializers.WritableDeviceBayTemplateSerializer
  140. filter_class = filters.DeviceBayTemplateFilter
  141. #
  142. # Device roles
  143. #
  144. class DeviceRoleViewSet(ModelViewSet):
  145. queryset = DeviceRole.objects.all()
  146. serializer_class = serializers.DeviceRoleSerializer
  147. #
  148. # Platforms
  149. #
  150. class PlatformViewSet(ModelViewSet):
  151. queryset = Platform.objects.all()
  152. serializer_class = serializers.PlatformSerializer
  153. #
  154. # Devices
  155. #
  156. class DeviceViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
  157. queryset = Device.objects.select_related(
  158. 'device_type__manufacturer', 'device_role', 'tenant', 'platform', 'rack__site', 'parent_bay',
  159. ).prefetch_related(
  160. 'primary_ip4__nat_outside', 'primary_ip6__nat_outside',
  161. )
  162. serializer_class = serializers.DeviceSerializer
  163. write_serializer_class = serializers.WritableDeviceSerializer
  164. filter_class = filters.DeviceFilter
  165. renderer_classes = api_settings.DEFAULT_RENDERER_CLASSES + [BINDZoneRenderer, FlatJSONRenderer]
  166. @detail_route(url_path='lldp-neighbors')
  167. def lldp_neighbors(self, request, pk):
  168. """
  169. Retrieve live LLDP neighbors of a device
  170. """
  171. device = get_object_or_404(Device, pk=pk)
  172. if not device.primary_ip:
  173. raise ServiceUnavailable("No IP configured for this device.")
  174. RPC = device.get_rpc_client()
  175. if not RPC:
  176. raise ServiceUnavailable("No RPC client available for this platform ({}).".format(device.platform))
  177. # Connect to device and retrieve inventory info
  178. try:
  179. with RPC(device, username=settings.NETBOX_USERNAME, password=settings.NETBOX_PASSWORD) as rpc_client:
  180. lldp_neighbors = rpc_client.get_lldp_neighbors()
  181. except:
  182. raise ServiceUnavailable("Error connecting to the remote device.")
  183. return Response(lldp_neighbors)
  184. #
  185. # Device components
  186. #
  187. class ConsolePortViewSet(WritableSerializerMixin, ModelViewSet):
  188. queryset = ConsolePort.objects.select_related('device', 'cs_port__device')
  189. serializer_class = serializers.ConsolePortSerializer
  190. write_serializer_class = serializers.WritableConsolePortSerializer
  191. filter_class = filters.ConsolePortFilter
  192. class ConsoleServerPortViewSet(WritableSerializerMixin, ModelViewSet):
  193. queryset = ConsoleServerPort.objects.select_related('device', 'connected_console__device')
  194. serializer_class = serializers.ConsoleServerPortSerializer
  195. write_serializer_class = serializers.WritableConsoleServerPortSerializer
  196. filter_class = filters.ConsoleServerPortFilter
  197. class PowerPortViewSet(WritableSerializerMixin, ModelViewSet):
  198. queryset = PowerPort.objects.select_related('device', 'power_outlet__device')
  199. serializer_class = serializers.PowerPortSerializer
  200. write_serializer_class = serializers.WritablePowerPortSerializer
  201. filter_class = filters.PowerPortFilter
  202. class PowerOutletViewSet(WritableSerializerMixin, ModelViewSet):
  203. queryset = PowerOutlet.objects.select_related('device', 'connected_port__device')
  204. serializer_class = serializers.PowerOutletSerializer
  205. write_serializer_class = serializers.WritablePowerOutletSerializer
  206. filter_class = filters.PowerOutletFilter
  207. class InterfaceViewSet(WritableSerializerMixin, ModelViewSet):
  208. queryset = Interface.objects.select_related('device')
  209. serializer_class = serializers.InterfaceSerializer
  210. write_serializer_class = serializers.WritableInterfaceSerializer
  211. filter_class = filters.InterfaceFilter
  212. @detail_route()
  213. def graphs(self, request, pk=None):
  214. interface = get_object_or_404(Interface, pk=pk)
  215. queryset = Graph.objects.filter(type=GRAPH_TYPE_INTERFACE)
  216. serializer = GraphSerializer(queryset, many=True, context={'graphed_object': interface})
  217. return Response(serializer.data)
  218. class DeviceBayViewSet(WritableSerializerMixin, ModelViewSet):
  219. queryset = DeviceBay.objects.select_related('installed_device')
  220. serializer_class = serializers.DeviceBaySerializer
  221. write_serializer_class = serializers.WritableDeviceBaySerializer
  222. filter_class = filters.DeviceBayFilter
  223. class ModuleViewSet(WritableSerializerMixin, ModelViewSet):
  224. queryset = Module.objects.select_related('device', 'manufacturer')
  225. serializer_class = serializers.ModuleSerializer
  226. write_serializer_class = serializers.WritableModuleSerializer
  227. filter_class = filters.ModuleFilter
  228. #
  229. # Interface connections
  230. #
  231. class InterfaceConnectionViewSet(WritableSerializerMixin, ModelViewSet):
  232. queryset = InterfaceConnection.objects.select_related('interface_a__device', 'interface_b__device')
  233. serializer_class = serializers.InterfaceConnectionSerializer
  234. write_serializer_class = serializers.WritableInterfaceConnectionSerializer
  235. #
  236. # Live queries
  237. #
  238. class LLDPNeighborsView(APIView):
  239. """
  240. Retrieve live LLDP neighbors of a device
  241. """
  242. def get(self, request, pk):
  243. device = get_object_or_404(Device, pk=pk)
  244. if not device.primary_ip:
  245. raise ServiceUnavailable(detail="No IP configured for this device.")
  246. RPC = device.get_rpc_client()
  247. if not RPC:
  248. raise ServiceUnavailable(detail="No RPC client available for this platform ({}).".format(device.platform))
  249. # Connect to device and retrieve inventory info
  250. try:
  251. with RPC(device, username=settings.NETBOX_USERNAME, password=settings.NETBOX_PASSWORD) as rpc_client:
  252. lldp_neighbors = rpc_client.get_lldp_neighbors()
  253. except:
  254. raise ServiceUnavailable(detail="Error connecting to the remote device.")
  255. return Response(lldp_neighbors)
  256. #
  257. # Miscellaneous
  258. #
  259. class RelatedConnectionsView(APIView):
  260. """
  261. Retrieve all connections related to a given console/power/interface connection
  262. """
  263. def __init__(self):
  264. super(RelatedConnectionsView, self).__init__()
  265. # Custom fields
  266. content_type = ContentType.objects.get_for_model(Device)
  267. custom_fields = content_type.custom_fields.prefetch_related('choices')
  268. # Cache all relevant CustomFieldChoices. This saves us from having to do a lookup per select field per object.
  269. custom_field_choices = {}
  270. for field in custom_fields:
  271. for cfc in field.choices.all():
  272. custom_field_choices[cfc.id] = cfc.value
  273. self.context = {
  274. 'custom_fields': custom_fields,
  275. 'custom_field_choices': custom_field_choices,
  276. }
  277. def get(self, request):
  278. peer_device = request.GET.get('peer-device')
  279. peer_interface = request.GET.get('peer-interface')
  280. # Search by interface
  281. if peer_device and peer_interface:
  282. # Determine local interface from peer interface's connection
  283. try:
  284. peer_iface = Interface.objects.get(device__name=peer_device, name=peer_interface)
  285. except Interface.DoesNotExist:
  286. raise Http404()
  287. local_iface = peer_iface.connected_interface
  288. if local_iface:
  289. device = local_iface.device
  290. else:
  291. return Response()
  292. else:
  293. raise MissingFilterException(detail='Must specify search parameters "peer-device" and "peer-interface".')
  294. # Initialize response skeleton
  295. response = {
  296. 'device': serializers.DeviceSerializer(device, context=self.context).data,
  297. 'console-ports': [],
  298. 'power-ports': [],
  299. 'interfaces': [],
  300. }
  301. # Console connections
  302. console_ports = ConsolePort.objects.filter(device=device).select_related('cs_port__device')
  303. for cp in console_ports:
  304. data = serializers.ConsolePortSerializer(instance=cp).data
  305. del(data['device'])
  306. response['console-ports'].append(data)
  307. # Power connections
  308. power_ports = PowerPort.objects.filter(device=device).select_related('power_outlet__device')
  309. for pp in power_ports:
  310. data = serializers.PowerPortSerializer(instance=pp).data
  311. del(data['device'])
  312. response['power-ports'].append(data)
  313. # Interface connections
  314. interfaces = Interface.objects.order_naturally(device.device_type.interface_ordering).filter(device=device)\
  315. .select_related('connected_as_a', 'connected_as_b', 'circuit_termination')
  316. for iface in interfaces:
  317. data = serializers.InterfaceSerializer(instance=iface).data
  318. del(data['device'])
  319. response['interfaces'].append(data)
  320. return Response(response)