123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493 |
- from rest_framework import generics
- from rest_framework.permissions import DjangoModelPermissionsOrAnonReadOnly
- from rest_framework.response import Response
- from rest_framework.settings import api_settings
- from rest_framework.views import APIView
- from django.conf import settings
- from django.contrib.contenttypes.models import ContentType
- from django.http import Http404
- from django.shortcuts import get_object_or_404
- from dcim.models import (
- ConsolePort, ConsoleServerPort, Device, DeviceBay, DeviceRole, DeviceType, IFACE_FF_VIRTUAL, Interface,
- InterfaceConnection, Manufacturer, Module, Platform, PowerOutlet, PowerPort, Rack, RackGroup, RackRole, Site,
- )
- from dcim import filters
- from extras.api.views import CustomFieldModelAPIView
- from extras.api.renderers import BINDZoneRenderer, FlatJSONRenderer
- from utilities.api import ServiceUnavailable
- from .exceptions import MissingFilterException
- from . import serializers
- #
- # Sites
- #
- class SiteListView(CustomFieldModelAPIView, generics.ListAPIView):
- """
- List all sites
- """
- queryset = Site.objects.select_related('tenant').prefetch_related('custom_field_values__field')
- serializer_class = serializers.SiteSerializer
- class SiteDetailView(CustomFieldModelAPIView, generics.RetrieveAPIView):
- """
- Retrieve a single site
- """
- queryset = Site.objects.select_related('tenant').prefetch_related('custom_field_values__field')
- serializer_class = serializers.SiteSerializer
- #
- # Rack groups
- #
- class RackGroupListView(generics.ListAPIView):
- """
- List all rack groups
- """
- queryset = RackGroup.objects.select_related('site')
- serializer_class = serializers.RackGroupSerializer
- filter_class = filters.RackGroupFilter
- class RackGroupDetailView(generics.RetrieveAPIView):
- """
- Retrieve a single rack group
- """
- queryset = RackGroup.objects.select_related('site')
- serializer_class = serializers.RackGroupSerializer
- #
- # Rack roles
- #
- class RackRoleListView(generics.ListAPIView):
- """
- List all rack roles
- """
- queryset = RackRole.objects.all()
- serializer_class = serializers.RackRoleSerializer
- class RackRoleDetailView(generics.RetrieveAPIView):
- """
- Retrieve a single rack role
- """
- queryset = RackRole.objects.all()
- serializer_class = serializers.RackRoleSerializer
- #
- # Racks
- #
- class RackListView(CustomFieldModelAPIView, generics.ListAPIView):
- """
- List racks (filterable)
- """
- queryset = Rack.objects.select_related('site', 'group__site', 'tenant')\
- .prefetch_related('custom_field_values__field')
- serializer_class = serializers.RackSerializer
- filter_class = filters.RackFilter
- class RackDetailView(CustomFieldModelAPIView, generics.RetrieveAPIView):
- """
- Retrieve a single rack
- """
- queryset = Rack.objects.select_related('site', 'group__site', 'tenant')\
- .prefetch_related('custom_field_values__field')
- serializer_class = serializers.RackDetailSerializer
- #
- # Rack units
- #
- class RackUnitListView(APIView):
- """
- List rack units (by rack)
- """
- def get(self, request, pk):
- rack = get_object_or_404(Rack, pk=pk)
- face = request.GET.get('face', 0)
- elevation = rack.get_rack_units(face)
- # Serialize Devices within the rack elevation
- for u in elevation:
- if u['device']:
- u['device'] = serializers.DeviceNestedSerializer(instance=u['device']).data
- return Response(elevation)
- #
- # Manufacturers
- #
- class ManufacturerListView(generics.ListAPIView):
- """
- List all hardware manufacturers
- """
- queryset = Manufacturer.objects.all()
- serializer_class = serializers.ManufacturerSerializer
- class ManufacturerDetailView(generics.RetrieveAPIView):
- """
- Retrieve a single hardware manufacturers
- """
- queryset = Manufacturer.objects.all()
- serializer_class = serializers.ManufacturerSerializer
- #
- # Device Types
- #
- class DeviceTypeListView(generics.ListAPIView):
- """
- List device types (filterable)
- """
- queryset = DeviceType.objects.select_related('manufacturer')
- serializer_class = serializers.DeviceTypeSerializer
- filter_class = filters.DeviceTypeFilter
- class DeviceTypeDetailView(generics.RetrieveAPIView):
- """
- Retrieve a single device type
- """
- queryset = DeviceType.objects.select_related('manufacturer')
- serializer_class = serializers.DeviceTypeDetailSerializer
- #
- # Device roles
- #
- class DeviceRoleListView(generics.ListAPIView):
- """
- List all device roles
- """
- queryset = DeviceRole.objects.all()
- serializer_class = serializers.DeviceRoleSerializer
- class DeviceRoleDetailView(generics.RetrieveAPIView):
- """
- Retrieve a single device role
- """
- queryset = DeviceRole.objects.all()
- serializer_class = serializers.DeviceRoleSerializer
- #
- # Platforms
- #
- class PlatformListView(generics.ListAPIView):
- """
- List all platforms
- """
- queryset = Platform.objects.all()
- serializer_class = serializers.PlatformSerializer
- class PlatformDetailView(generics.RetrieveAPIView):
- """
- Retrieve a single platform
- """
- queryset = Platform.objects.all()
- serializer_class = serializers.PlatformSerializer
- #
- # Devices
- #
- class DeviceListView(CustomFieldModelAPIView, generics.ListAPIView):
- """
- List devices (filterable)
- """
- queryset = Device.objects.select_related('device_type__manufacturer', 'device_role', 'tenant', 'platform',
- 'rack__site', 'parent_bay').prefetch_related('primary_ip4__nat_outside',
- 'primary_ip6__nat_outside',
- 'custom_field_values__field')
- serializer_class = serializers.DeviceSerializer
- filter_class = filters.DeviceFilter
- renderer_classes = api_settings.DEFAULT_RENDERER_CLASSES + [BINDZoneRenderer, FlatJSONRenderer]
- class DeviceDetailView(CustomFieldModelAPIView, generics.RetrieveAPIView):
- """
- Retrieve a single device
- """
- queryset = Device.objects.select_related('device_type__manufacturer', 'device_role', 'tenant', 'platform',
- 'rack__site', 'parent_bay').prefetch_related('custom_field_values__field')
- serializer_class = serializers.DeviceSerializer
- #
- # Console ports
- #
- class ConsolePortListView(generics.ListAPIView):
- """
- List console ports (by device)
- """
- serializer_class = serializers.ConsolePortSerializer
- def get_queryset(self):
- device = get_object_or_404(Device, pk=self.kwargs['pk'])
- return ConsolePort.objects.filter(device=device).select_related('cs_port')
- class ConsolePortView(generics.RetrieveUpdateDestroyAPIView):
- permission_classes = [DjangoModelPermissionsOrAnonReadOnly]
- serializer_class = serializers.ConsolePortSerializer
- queryset = ConsolePort.objects.all()
- #
- # Console server ports
- #
- class ConsoleServerPortListView(generics.ListAPIView):
- """
- List console server ports (by device)
- """
- serializer_class = serializers.ConsoleServerPortSerializer
- def get_queryset(self):
- device = get_object_or_404(Device, pk=self.kwargs['pk'])
- return ConsoleServerPort.objects.filter(device=device).select_related('connected_console')
- #
- # Power ports
- #
- class PowerPortListView(generics.ListAPIView):
- """
- List power ports (by device)
- """
- serializer_class = serializers.PowerPortSerializer
- def get_queryset(self):
- device = get_object_or_404(Device, pk=self.kwargs['pk'])
- return PowerPort.objects.filter(device=device).select_related('power_outlet')
- class PowerPortView(generics.RetrieveUpdateDestroyAPIView):
- permission_classes = [DjangoModelPermissionsOrAnonReadOnly]
- serializer_class = serializers.PowerPortSerializer
- queryset = PowerPort.objects.all()
- #
- # Power outlets
- #
- class PowerOutletListView(generics.ListAPIView):
- """
- List power outlets (by device)
- """
- serializer_class = serializers.PowerOutletSerializer
- def get_queryset(self):
- device = get_object_or_404(Device, pk=self.kwargs['pk'])
- return PowerOutlet.objects.filter(device=device).select_related('connected_port')
- #
- # Interfaces
- #
- class InterfaceListView(generics.ListAPIView):
- """
- List interfaces (by device)
- """
- serializer_class = serializers.InterfaceSerializer
- filter_class = filters.InterfaceFilter
- def get_queryset(self):
- device = get_object_or_404(Device, pk=self.kwargs['pk'])
- queryset = Interface.objects.filter(device=device).select_related('connected_as_a', 'connected_as_b')
- # Filter by type (physical or virtual)
- iface_type = self.request.query_params.get('type')
- if iface_type == 'physical':
- queryset = queryset.exclude(form_factor=IFACE_FF_VIRTUAL)
- elif iface_type == 'virtual':
- queryset = queryset.filter(form_factor=IFACE_FF_VIRTUAL)
- elif iface_type is not None:
- queryset = queryset.empty()
- return queryset
- class InterfaceDetailView(generics.RetrieveAPIView):
- """
- Retrieve a single interface
- """
- queryset = Interface.objects.select_related('device')
- serializer_class = serializers.InterfaceDetailSerializer
- class InterfaceConnectionView(generics.RetrieveUpdateDestroyAPIView):
- permission_classes = [DjangoModelPermissionsOrAnonReadOnly]
- serializer_class = serializers.InterfaceConnectionSerializer
- queryset = InterfaceConnection.objects.all()
- class InterfaceConnectionListView(generics.ListAPIView):
- """
- Retrieve a list of all interface connections
- """
- serializer_class = serializers.InterfaceConnectionSerializer
- queryset = InterfaceConnection.objects.all()
- #
- # Device bays
- #
- class DeviceBayListView(generics.ListAPIView):
- """
- List device bays (by device)
- """
- serializer_class = serializers.DeviceBayNestedSerializer
- def get_queryset(self):
- device = get_object_or_404(Device, pk=self.kwargs['pk'])
- return DeviceBay.objects.filter(device=device).select_related('installed_device')
- #
- # Modules
- #
- class ModuleListView(generics.ListAPIView):
- """
- List device modules (by device)
- """
- serializer_class = serializers.ModuleSerializer
- def get_queryset(self):
- device = get_object_or_404(Device, pk=self.kwargs['pk'])
- return Module.objects.filter(device=device).select_related('device', 'manufacturer')
- #
- # Live queries
- #
- class LLDPNeighborsView(APIView):
- """
- Retrieve live LLDP neighbors of a device
- """
- def get(self, request, pk):
- device = get_object_or_404(Device, pk=pk)
- if not device.primary_ip:
- raise ServiceUnavailable(detail="No IP configured for this device.")
- RPC = device.get_rpc_client()
- if not RPC:
- raise ServiceUnavailable(detail="No RPC client available for this platform ({}).".format(device.platform))
- # Connect to device and retrieve inventory info
- try:
- with RPC(device, username=settings.NETBOX_USERNAME, password=settings.NETBOX_PASSWORD) as rpc_client:
- lldp_neighbors = rpc_client.get_lldp_neighbors()
- except:
- raise ServiceUnavailable(detail="Error connecting to the remote device.")
- return Response(lldp_neighbors)
- #
- # Miscellaneous
- #
- class RelatedConnectionsView(APIView):
- """
- Retrieve all connections related to a given console/power/interface connection
- """
- def __init__(self):
- super(RelatedConnectionsView, self).__init__()
- # Custom fields
- self.content_type = ContentType.objects.get_for_model(Device)
- self.custom_fields = self.content_type.custom_fields.prefetch_related('choices')
- def get(self, request):
- peer_device = request.GET.get('peer-device')
- peer_interface = request.GET.get('peer-interface')
- # Search by interface
- if peer_device and peer_interface:
- # Determine local interface from peer interface's connection
- try:
- peer_iface = Interface.objects.get(device__name=peer_device, name=peer_interface)
- except Interface.DoesNotExist:
- raise Http404()
- local_iface = peer_iface.get_connected_interface()
- if local_iface:
- device = local_iface.device
- else:
- return Response()
- else:
- raise MissingFilterException(detail='Must specify search parameters "peer-device" and "peer-interface".')
- # Initialize response skeleton
- response = {
- 'device': serializers.DeviceSerializer(device, context={'view': self}).data,
- 'console-ports': [],
- 'power-ports': [],
- 'interfaces': [],
- }
- # Console connections
- console_ports = ConsolePort.objects.filter(device=device).select_related('cs_port__device')
- for cp in console_ports:
- data = serializers.ConsolePortSerializer(instance=cp).data
- del(data['device'])
- response['console-ports'].append(data)
- # Power connections
- power_ports = PowerPort.objects.filter(device=device).select_related('power_outlet__device')
- for pp in power_ports:
- data = serializers.PowerPortSerializer(instance=pp).data
- del(data['device'])
- response['power-ports'].append(data)
- # Interface connections
- interfaces = Interface.objects.filter(device=device).select_related('connected_as_a', 'connected_as_b',
- 'circuit')
- for iface in interfaces:
- data = serializers.InterfaceDetailSerializer(instance=iface).data
- del(data['device'])
- response['interfaces'].append(data)
- return Response(response)
|