1
0
mirror of https://github.com/netbox-community/netbox.git synced 2024-05-10 07:54:54 +00:00

364 lines
13 KiB
Python
Raw Normal View History

2017-05-24 11:33:11 -04:00
from __future__ import unicode_literals
from rest_framework.decorators import detail_route
from rest_framework.mixins import ListModelMixin
from rest_framework.permissions import IsAuthenticated
2016-03-01 11:23:03 -05:00
from rest_framework.response import Response
from rest_framework.viewsets import GenericViewSet, ModelViewSet, ViewSet
2016-03-01 11:23:03 -05:00
from django.conf import settings
from django.shortcuts import get_object_or_404
2016-05-18 16:02:53 -04:00
from dcim.models import (
ConsolePort, ConsolePortTemplate, ConsoleServerPort, ConsoleServerPortTemplate, Device, DeviceBay,
DeviceBayTemplate, DeviceRole, DeviceType, Interface, InterfaceConnection, InterfaceTemplate, Manufacturer,
InventoryItem, Platform, PowerOutlet, PowerOutletTemplate, PowerPort, PowerPortTemplate, Rack, RackGroup,
RackReservation, RackRole, Region, Site,
2016-05-18 16:02:53 -04:00
)
from dcim import filters
2017-03-20 15:14:33 -04:00
from extras.api.serializers import RenderedGraphSerializer
2017-01-30 17:00:58 -05:00
from extras.api.views import CustomFieldModelViewSet
from extras.models import Graph, GRAPH_TYPE_INTERFACE, GRAPH_TYPE_SITE
2017-01-27 14:36:13 -05:00
from utilities.api import ServiceUnavailable, WritableSerializerMixin
2016-08-22 13:20:30 -04:00
from .exceptions import MissingFilterException
from . import serializers
2016-03-01 11:23:03 -05:00
2017-02-28 16:10:53 -05:00
#
# Regions
#
2017-02-28 16:21:31 -05:00
class RegionViewSet(WritableSerializerMixin, ModelViewSet):
2017-02-28 16:10:53 -05:00
queryset = Region.objects.all()
serializer_class = serializers.RegionSerializer
write_serializer_class = serializers.WritableRegionSerializer
2016-03-01 11:23:03 -05:00
#
# Sites
#
2017-01-27 14:36:13 -05:00
class SiteViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
2017-02-28 16:21:31 -05:00
queryset = Site.objects.select_related('region', 'tenant')
2016-05-18 16:02:53 -04:00
serializer_class = serializers.SiteSerializer
write_serializer_class = serializers.WritableSiteSerializer
filter_class = filters.SiteFilter
2016-03-01 11:23:03 -05:00
2017-01-30 17:00:58 -05:00
@detail_route()
def graphs(self, request, pk=None):
2017-03-20 15:14:33 -04:00
"""
A convenience method for rendering graphs for a particular site.
"""
2017-01-30 17:00:58 -05:00
site = get_object_or_404(Site, pk=pk)
queryset = Graph.objects.filter(type=GRAPH_TYPE_SITE)
2017-03-20 15:14:33 -04:00
serializer = RenderedGraphSerializer(queryset, many=True, context={'graphed_object': site})
2017-01-30 17:00:58 -05:00
return Response(serializer.data)
2016-03-01 11:23:03 -05:00
#
# Rack groups
#
2017-01-27 14:36:13 -05:00
class RackGroupViewSet(WritableSerializerMixin, ModelViewSet):
2016-07-27 14:40:19 -04:00
queryset = RackGroup.objects.select_related('site')
2016-05-18 16:02:53 -04:00
serializer_class = serializers.RackGroupSerializer
write_serializer_class = serializers.WritableRackGroupSerializer
filter_class = filters.RackGroupFilter
2016-03-01 11:23:03 -05:00
2016-08-10 11:52:27 -04:00
#
# Rack roles
#
2017-01-24 17:12:16 -05:00
class RackRoleViewSet(ModelViewSet):
2016-08-10 11:52:27 -04:00
queryset = RackRole.objects.all()
serializer_class = serializers.RackRoleSerializer
2016-03-01 11:23:03 -05:00
#
# Racks
#
2017-01-27 14:36:13 -05:00
class RackViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
2017-01-24 17:12:16 -05:00
queryset = Rack.objects.select_related('site', 'group__site', 'tenant')
serializer_class = serializers.RackSerializer
write_serializer_class = serializers.WritableRackSerializer
2016-05-18 16:02:53 -04:00
filter_class = filters.RackFilter
2016-03-01 11:23:03 -05:00
@detail_route()
def units(self, request, pk=None):
"""
List rack units (by rack)
"""
2016-03-01 11:23:03 -05:00
rack = get_object_or_404(Rack, pk=pk)
face = request.GET.get('face', 0)
exclude_pk = request.GET.get('exclude', None)
if exclude_pk is not None:
try:
exclude_pk = int(exclude_pk)
except ValueError:
exclude_pk = None
elevation = rack.get_rack_units(face, exclude_pk)
2016-03-01 11:23:03 -05:00
page = self.paginate_queryset(elevation)
if page is not None:
rack_units = serializers.RackUnitSerializer(page, many=True, context={'request': request})
return self.get_paginated_response(rack_units.data)
2016-03-01 11:23:03 -05:00
#
# Rack reservations
#
2017-03-16 16:50:18 -04:00
class RackReservationViewSet(WritableSerializerMixin, ModelViewSet):
queryset = RackReservation.objects.select_related('rack')
serializer_class = serializers.RackReservationSerializer
2017-03-16 16:50:18 -04:00
write_serializer_class = serializers.WritableRackReservationSerializer
filter_class = filters.RackReservationFilter
# Assign user from request
def perform_create(self, serializer):
serializer.save(user=self.request.user)
2016-03-01 11:23:03 -05:00
#
# Manufacturers
#
2017-01-24 17:12:16 -05:00
class ManufacturerViewSet(ModelViewSet):
2016-03-01 11:23:03 -05:00
queryset = Manufacturer.objects.all()
2016-05-18 16:02:53 -04:00
serializer_class = serializers.ManufacturerSerializer
2016-03-01 11:23:03 -05:00
#
# Device types
2016-03-01 11:23:03 -05:00
#
2017-01-27 14:36:13 -05:00
class DeviceTypeViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
2017-01-24 17:12:16 -05:00
queryset = DeviceType.objects.select_related('manufacturer')
serializer_class = serializers.DeviceTypeSerializer
write_serializer_class = serializers.WritableDeviceTypeSerializer
filter_class = filters.DeviceTypeFilter
2016-03-01 11:23:03 -05:00
#
# Device type components
#
class ConsolePortTemplateViewSet(WritableSerializerMixin, ModelViewSet):
queryset = ConsolePortTemplate.objects.select_related('device_type__manufacturer')
serializer_class = serializers.ConsolePortTemplateSerializer
write_serializer_class = serializers.WritableConsolePortTemplateSerializer
filter_class = filters.ConsolePortTemplateFilter
class ConsoleServerPortTemplateViewSet(WritableSerializerMixin, ModelViewSet):
queryset = ConsoleServerPortTemplate.objects.select_related('device_type__manufacturer')
serializer_class = serializers.ConsoleServerPortTemplateSerializer
write_serializer_class = serializers.WritableConsoleServerPortTemplateSerializer
filter_class = filters.ConsoleServerPortTemplateFilter
class PowerPortTemplateViewSet(WritableSerializerMixin, ModelViewSet):
queryset = PowerPortTemplate.objects.select_related('device_type__manufacturer')
serializer_class = serializers.PowerPortTemplateSerializer
write_serializer_class = serializers.WritablePowerPortTemplateSerializer
filter_class = filters.PowerPortTemplateFilter
class PowerOutletTemplateViewSet(WritableSerializerMixin, ModelViewSet):
queryset = PowerOutletTemplate.objects.select_related('device_type__manufacturer')
serializer_class = serializers.PowerOutletTemplateSerializer
write_serializer_class = serializers.WritablePowerOutletTemplateSerializer
filter_class = filters.PowerOutletTemplateFilter
class InterfaceTemplateViewSet(WritableSerializerMixin, ModelViewSet):
queryset = InterfaceTemplate.objects.select_related('device_type__manufacturer')
serializer_class = serializers.InterfaceTemplateSerializer
write_serializer_class = serializers.WritableInterfaceTemplateSerializer
filter_class = filters.InterfaceTemplateFilter
class DeviceBayTemplateViewSet(WritableSerializerMixin, ModelViewSet):
queryset = DeviceBayTemplate.objects.select_related('device_type__manufacturer')
serializer_class = serializers.DeviceBayTemplateSerializer
write_serializer_class = serializers.WritableDeviceBayTemplateSerializer
filter_class = filters.DeviceBayTemplateFilter
#
# Device roles
2016-03-01 11:23:03 -05:00
#
2017-01-24 17:12:16 -05:00
class DeviceRoleViewSet(ModelViewSet):
2016-03-01 11:23:03 -05:00
queryset = DeviceRole.objects.all()
2016-05-18 16:02:53 -04:00
serializer_class = serializers.DeviceRoleSerializer
2016-03-01 11:23:03 -05:00
#
# Platforms
#
2017-01-24 17:12:16 -05:00
class PlatformViewSet(ModelViewSet):
2016-03-01 11:23:03 -05:00
queryset = Platform.objects.all()
2016-05-18 16:02:53 -04:00
serializer_class = serializers.PlatformSerializer
2016-03-01 11:23:03 -05:00
#
# Devices
#
2017-01-27 14:36:13 -05:00
class DeviceViewSet(WritableSerializerMixin, CustomFieldModelViewSet):
2017-01-24 17:12:16 -05:00
queryset = Device.objects.select_related(
2017-03-08 16:12:14 -05:00
'device_type__manufacturer', 'device_role', 'tenant', 'platform', 'site', 'rack', 'parent_bay',
2017-01-24 17:12:16 -05:00
).prefetch_related(
'primary_ip4__nat_outside', 'primary_ip6__nat_outside',
)
2016-05-18 16:02:53 -04:00
serializer_class = serializers.DeviceSerializer
write_serializer_class = serializers.WritableDeviceSerializer
2016-05-18 16:02:53 -04:00
filter_class = filters.DeviceFilter
2016-03-01 11:23:03 -05:00
@detail_route(url_path='lldp-neighbors')
def lldp_neighbors(self, request, pk):
"""
Retrieve live LLDP neighbors of a device
"""
device = get_object_or_404(Device, pk=pk)
if not device.primary_ip:
raise ServiceUnavailable("No IP configured for this device.")
RPC = device.get_rpc_client()
if not RPC:
raise ServiceUnavailable("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("Error connecting to the remote device.")
return Response(lldp_neighbors)
2016-03-01 11:23:03 -05:00
#
# Device components
2016-03-01 11:23:03 -05:00
#
class ConsolePortViewSet(WritableSerializerMixin, ModelViewSet):
queryset = ConsolePort.objects.select_related('device', 'cs_port__device')
serializer_class = serializers.ConsolePortSerializer
write_serializer_class = serializers.WritableConsolePortSerializer
filter_class = filters.ConsolePortFilter
class ConsoleServerPortViewSet(WritableSerializerMixin, ModelViewSet):
queryset = ConsoleServerPort.objects.select_related('device', 'connected_console__device')
serializer_class = serializers.ConsoleServerPortSerializer
write_serializer_class = serializers.WritableConsoleServerPortSerializer
filter_class = filters.ConsoleServerPortFilter
class PowerPortViewSet(WritableSerializerMixin, ModelViewSet):
queryset = PowerPort.objects.select_related('device', 'power_outlet__device')
serializer_class = serializers.PowerPortSerializer
write_serializer_class = serializers.WritablePowerPortSerializer
filter_class = filters.PowerPortFilter
class PowerOutletViewSet(WritableSerializerMixin, ModelViewSet):
queryset = PowerOutlet.objects.select_related('device', 'connected_port__device')
serializer_class = serializers.PowerOutletSerializer
write_serializer_class = serializers.WritablePowerOutletSerializer
filter_class = filters.PowerOutletFilter
class InterfaceViewSet(WritableSerializerMixin, ModelViewSet):
queryset = Interface.objects.select_related('device')
serializer_class = serializers.InterfaceSerializer
write_serializer_class = serializers.WritableInterfaceSerializer
filter_class = filters.InterfaceFilter
2017-01-30 17:00:58 -05:00
@detail_route()
def graphs(self, request, pk=None):
2017-03-20 15:14:33 -04:00
"""
A convenience method for rendering graphs for a particular interface.
"""
2017-01-30 17:00:58 -05:00
interface = get_object_or_404(Interface, pk=pk)
queryset = Graph.objects.filter(type=GRAPH_TYPE_INTERFACE)
2017-03-20 15:14:33 -04:00
serializer = RenderedGraphSerializer(queryset, many=True, context={'graphed_object': interface})
2017-01-30 17:00:58 -05:00
return Response(serializer.data)
class DeviceBayViewSet(WritableSerializerMixin, ModelViewSet):
queryset = DeviceBay.objects.select_related('installed_device')
serializer_class = serializers.DeviceBaySerializer
write_serializer_class = serializers.WritableDeviceBaySerializer
filter_class = filters.DeviceBayFilter
class InventoryItemViewSet(WritableSerializerMixin, ModelViewSet):
queryset = InventoryItem.objects.select_related('device', 'manufacturer')
serializer_class = serializers.InventoryItemSerializer
write_serializer_class = serializers.WritableInventoryItemSerializer
filter_class = filters.InventoryItemFilter
2017-01-31 12:19:41 -05:00
2016-07-05 13:43:19 -04:00
#
# Connections
#
class ConsoleConnectionViewSet(ListModelMixin, GenericViewSet):
queryset = ConsolePort.objects.select_related('device', 'cs_port__device').filter(cs_port__isnull=False)
serializer_class = serializers.ConsolePortSerializer
filter_class = filters.ConsoleConnectionFilter
class PowerConnectionViewSet(ListModelMixin, GenericViewSet):
queryset = PowerPort.objects.select_related('device', 'power_outlet__device').filter(power_outlet__isnull=False)
serializer_class = serializers.PowerPortSerializer
filter_class = filters.PowerConnectionFilter
class InterfaceConnectionViewSet(WritableSerializerMixin, ModelViewSet):
2017-01-30 17:00:58 -05:00
queryset = InterfaceConnection.objects.select_related('interface_a__device', 'interface_b__device')
serializer_class = serializers.InterfaceConnectionSerializer
write_serializer_class = serializers.WritableInterfaceConnectionSerializer
filter_class = filters.InterfaceConnectionFilter
2016-03-01 11:23:03 -05:00
#
# Miscellaneous
#
class ConnectedDeviceViewSet(ViewSet):
2016-03-01 11:23:03 -05:00
"""
This endpoint allows a user to determine what device (if any) is connected to a given peer device and peer
interface. This is useful in a situation where a device boots with no configuration, but can detect its neighbors
via a protocol such as LLDP. Two query parameters must be included in the request:
2017-01-24 17:12:16 -05:00
* `peer-device`: The name of the peer device
* `peer-interface`: The name of the peer interface
"""
permission_classes = [IsAuthenticated]
2017-01-24 17:12:16 -05:00
2017-03-20 21:54:01 -04:00
def get_view_name(self):
return "Connected Device Locator"
def list(self, request):
2016-08-22 17:15:20 -04:00
peer_device_name = request.query_params.get('peer-device')
peer_interface_name = request.query_params.get('peer-interface')
if not peer_device_name or not peer_interface_name:
raise MissingFilterException(detail='Request must include "peer-device" and "peer-interface" filters.')
2016-03-01 11:23:03 -05:00
# Determine local interface from peer interface's connection
peer_interface = get_object_or_404(Interface, device__name=peer_device_name, name=peer_interface_name)
local_interface = peer_interface.connected_interface
2016-03-01 11:23:03 -05:00
if local_interface is None:
return Response()
2016-03-01 11:23:03 -05:00
return Response(serializers.DeviceSerializer(local_interface.device, context={'request': request}).data)