2020-07-10 15:12:25 -05:00
|
|
|
import socket
|
2017-07-14 16:07:28 -04:00
|
|
|
from collections import OrderedDict
|
2017-05-24 11:33:11 -04:00
|
|
|
|
2017-11-07 11:08:23 -05:00
|
|
|
from django.conf import settings
|
2020-01-15 09:59:44 -05:00
|
|
|
from django.http import HttpResponseForbidden, HttpResponse
|
|
|
|
from django.shortcuts import get_object_or_404
|
2018-03-16 16:48:08 -04:00
|
|
|
from drf_yasg import openapi
|
|
|
|
from drf_yasg.openapi import Parameter
|
|
|
|
from drf_yasg.utils import swagger_auto_schema
|
2018-06-29 11:48:21 -04:00
|
|
|
from rest_framework.decorators import action
|
2016-03-01 11:23:03 -05:00
|
|
|
from rest_framework.response import Response
|
2020-08-13 12:45:38 -04:00
|
|
|
from rest_framework.routers import APIRootView
|
2021-07-14 15:29:04 -04:00
|
|
|
from rest_framework.viewsets import ViewSet
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2019-04-19 16:09:22 -04:00
|
|
|
from circuits.models import Circuit
|
2021-04-29 16:38:56 -04:00
|
|
|
from dcim import filtersets
|
2021-03-08 13:28:53 -05:00
|
|
|
from dcim.models import *
|
2020-10-30 02:56:43 -04:00
|
|
|
from extras.api.views import ConfigContextQuerySetMixin, CustomFieldModelViewSet
|
2019-04-19 16:09:22 -04:00
|
|
|
from ipam.models import Prefix, VLAN
|
2020-10-13 15:54:23 -04:00
|
|
|
from netbox.api.views import ModelViewSet
|
|
|
|
from netbox.api.authentication import IsAuthenticatedOrLoginNotRequired
|
|
|
|
from netbox.api.exceptions import ServiceUnavailable
|
|
|
|
from netbox.api.metadata import ContentTypeMetadata
|
|
|
|
from utilities.api import get_serializer_for_model
|
2021-08-31 23:56:18 -07:00
|
|
|
from utilities.utils import count_related, decode_dict
|
2019-04-12 17:18:04 -04:00
|
|
|
from virtualization.models import VirtualMachine
|
2016-08-22 13:20:30 -04:00
|
|
|
from . import serializers
|
2017-11-07 11:08:23 -05:00
|
|
|
from .exceptions import MissingFilterException
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
|
2020-08-13 12:45:38 -04:00
|
|
|
class DCIMRootView(APIRootView):
|
|
|
|
"""
|
|
|
|
DCIM API root view
|
|
|
|
"""
|
|
|
|
def get_view_name(self):
|
|
|
|
return 'DCIM'
|
|
|
|
|
|
|
|
|
2018-10-29 15:43:41 -04:00
|
|
|
# Mixins
|
|
|
|
|
2020-10-02 14:54:16 -04:00
|
|
|
class PathEndpointMixin(object):
|
2018-10-29 15:43:41 -04:00
|
|
|
|
|
|
|
@action(detail=True, url_path='trace')
|
|
|
|
def trace(self, request, pk):
|
|
|
|
"""
|
|
|
|
Trace a complete cable path and return each segment as a three-tuple of (termination, cable, termination).
|
|
|
|
"""
|
2020-06-29 14:41:43 -04:00
|
|
|
obj = get_object_or_404(self.queryset, pk=pk)
|
2018-10-29 15:43:41 -04:00
|
|
|
|
|
|
|
# Initialize the path array
|
|
|
|
path = []
|
|
|
|
|
2021-07-13 15:38:34 -04:00
|
|
|
if request.GET.get('render', None) == 'svg':
|
|
|
|
# Render SVG
|
2021-07-16 17:06:27 -04:00
|
|
|
try:
|
|
|
|
width = min(int(request.GET.get('width')), 1600)
|
2021-07-16 17:09:52 -04:00
|
|
|
except (ValueError, TypeError):
|
2021-07-16 17:06:27 -04:00
|
|
|
width = None
|
2021-07-13 15:38:34 -04:00
|
|
|
drawing = obj.get_trace_svg(
|
2021-07-16 17:06:27 -04:00
|
|
|
base_url=request.build_absolute_uri('/'),
|
|
|
|
width=width
|
2021-07-13 15:38:34 -04:00
|
|
|
)
|
|
|
|
return HttpResponse(drawing.tostring(), content_type='image/svg+xml')
|
|
|
|
|
2020-10-02 14:54:16 -04:00
|
|
|
for near_end, cable, far_end in obj.trace():
|
2020-10-08 14:01:47 -04:00
|
|
|
if near_end is None:
|
|
|
|
# Split paths
|
|
|
|
break
|
2018-10-29 15:43:41 -04:00
|
|
|
|
|
|
|
# Serialize each object
|
|
|
|
serializer_a = get_serializer_for_model(near_end, prefix='Nested')
|
|
|
|
x = serializer_a(near_end, context={'request': request}).data
|
|
|
|
if cable is not None:
|
|
|
|
y = serializers.TracedCableSerializer(cable, context={'request': request}).data
|
|
|
|
else:
|
|
|
|
y = None
|
|
|
|
if far_end is not None:
|
|
|
|
serializer_b = get_serializer_for_model(far_end, prefix='Nested')
|
|
|
|
z = serializer_b(far_end, context={'request': request}).data
|
|
|
|
else:
|
|
|
|
z = None
|
|
|
|
|
|
|
|
path.append((x, y, z))
|
|
|
|
|
|
|
|
return Response(path)
|
|
|
|
|
|
|
|
|
2020-10-08 10:32:17 -04:00
|
|
|
class PassThroughPortMixin(object):
|
|
|
|
|
|
|
|
@action(detail=True, url_path='paths')
|
|
|
|
def paths(self, request, pk):
|
|
|
|
"""
|
|
|
|
Return all CablePaths which traverse a given pass-through port.
|
|
|
|
"""
|
|
|
|
obj = get_object_or_404(self.queryset, pk=pk)
|
|
|
|
cablepaths = CablePath.objects.filter(path__contains=obj).prefetch_related('origin', 'destination')
|
|
|
|
serializer = serializers.CablePathSerializer(cablepaths, context={'request': request}, many=True)
|
|
|
|
|
|
|
|
return Response(serializer.data)
|
|
|
|
|
|
|
|
|
2017-02-28 16:10:53 -05:00
|
|
|
#
|
|
|
|
# Regions
|
|
|
|
#
|
|
|
|
|
2021-02-25 15:58:13 -05:00
|
|
|
class RegionViewSet(CustomFieldModelViewSet):
|
2020-07-20 12:07:19 -04:00
|
|
|
queryset = Region.objects.add_related_count(
|
|
|
|
Region.objects.all(),
|
|
|
|
Site,
|
|
|
|
'region',
|
|
|
|
'site_count',
|
|
|
|
cumulative=True
|
2019-04-12 17:07:56 -04:00
|
|
|
)
|
2017-02-28 16:10:53 -05:00
|
|
|
serializer_class = serializers.RegionSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.RegionFilterSet
|
2017-02-28 16:10:53 -05:00
|
|
|
|
|
|
|
|
2021-03-08 13:28:53 -05:00
|
|
|
#
|
|
|
|
# Site groups
|
|
|
|
#
|
|
|
|
|
|
|
|
class SiteGroupViewSet(CustomFieldModelViewSet):
|
|
|
|
queryset = SiteGroup.objects.add_related_count(
|
|
|
|
SiteGroup.objects.all(),
|
|
|
|
Site,
|
|
|
|
'group',
|
|
|
|
'site_count',
|
|
|
|
cumulative=True
|
|
|
|
)
|
|
|
|
serializer_class = serializers.SiteGroupSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.SiteGroupFilterSet
|
2021-03-08 13:28:53 -05:00
|
|
|
|
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
#
|
|
|
|
# Sites
|
|
|
|
#
|
|
|
|
|
2017-11-07 15:36:10 -05:00
|
|
|
class SiteViewSet(CustomFieldModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = Site.objects.prefetch_related(
|
|
|
|
'region', 'tenant', 'tags'
|
2019-04-19 16:09:22 -04:00
|
|
|
).annotate(
|
2020-12-17 14:47:49 -05:00
|
|
|
device_count=count_related(Device, 'site'),
|
|
|
|
rack_count=count_related(Rack, 'site'),
|
|
|
|
prefix_count=count_related(Prefix, 'site'),
|
|
|
|
vlan_count=count_related(VLAN, 'site'),
|
|
|
|
circuit_count=count_related(Circuit, 'terminations__site'),
|
|
|
|
virtualmachine_count=count_related(VirtualMachine, 'cluster__site')
|
2020-11-25 15:49:18 -05:00
|
|
|
)
|
2016-05-18 16:02:53 -04:00
|
|
|
serializer_class = serializers.SiteSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.SiteFilterSet
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
|
|
|
|
#
|
2021-04-11 13:43:06 -04:00
|
|
|
# Locations
|
2016-03-01 11:23:03 -05:00
|
|
|
#
|
|
|
|
|
2021-03-03 13:30:33 -05:00
|
|
|
class LocationViewSet(CustomFieldModelViewSet):
|
|
|
|
queryset = Location.objects.add_related_count(
|
2021-04-05 11:04:12 -04:00
|
|
|
Location.objects.add_related_count(
|
|
|
|
Location.objects.all(),
|
|
|
|
Device,
|
|
|
|
'location',
|
|
|
|
'device_count',
|
|
|
|
cumulative=True
|
|
|
|
),
|
2020-07-20 12:07:19 -04:00
|
|
|
Rack,
|
2021-03-03 13:30:33 -05:00
|
|
|
'location',
|
2020-07-20 12:07:19 -04:00
|
|
|
'rack_count',
|
|
|
|
cumulative=True
|
|
|
|
).prefetch_related('site')
|
2021-03-03 13:30:33 -05:00
|
|
|
serializer_class = serializers.LocationSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.LocationFilterSet
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
|
2016-08-10 11:52:27 -04:00
|
|
|
#
|
|
|
|
# Rack roles
|
|
|
|
#
|
|
|
|
|
2021-02-25 15:58:13 -05:00
|
|
|
class RackRoleViewSet(CustomFieldModelViewSet):
|
2019-04-12 17:07:56 -04:00
|
|
|
queryset = RackRole.objects.annotate(
|
2020-12-17 14:47:49 -05:00
|
|
|
rack_count=count_related(Rack, 'role')
|
2020-11-25 15:49:18 -05:00
|
|
|
)
|
2016-08-10 11:52:27 -04:00
|
|
|
serializer_class = serializers.RackRoleSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.RackRoleFilterSet
|
2016-08-10 11:52:27 -04:00
|
|
|
|
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
#
|
|
|
|
# Racks
|
|
|
|
#
|
|
|
|
|
2017-11-07 15:36:10 -05:00
|
|
|
class RackViewSet(CustomFieldModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = Rack.objects.prefetch_related(
|
2021-04-05 11:04:12 -04:00
|
|
|
'site', 'location', 'role', 'tenant', 'tags'
|
2019-04-12 17:07:56 -04:00
|
|
|
).annotate(
|
2020-12-17 14:47:49 -05:00
|
|
|
device_count=count_related(Device, 'rack'),
|
|
|
|
powerfeed_count=count_related(PowerFeed, 'rack')
|
2020-11-25 15:49:18 -05:00
|
|
|
)
|
2017-01-27 12:22:29 -05:00
|
|
|
serializer_class = serializers.RackSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.RackFilterSet
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2019-12-11 09:45:08 -05:00
|
|
|
@swagger_auto_schema(
|
|
|
|
responses={200: serializers.RackUnitSerializer(many=True)},
|
|
|
|
query_serializer=serializers.RackElevationDetailFilterSerializer
|
|
|
|
)
|
2019-12-10 03:18:10 -05:00
|
|
|
@action(detail=True)
|
|
|
|
def elevation(self, request, pk=None):
|
|
|
|
"""
|
|
|
|
Rack elevation representing the list of rack units. Also supports rendering the elevation as an SVG.
|
|
|
|
"""
|
2020-06-29 14:41:43 -04:00
|
|
|
rack = get_object_or_404(self.queryset, pk=pk)
|
2019-12-11 09:45:08 -05:00
|
|
|
serializer = serializers.RackElevationDetailFilterSerializer(data=request.GET)
|
|
|
|
if not serializer.is_valid():
|
|
|
|
return Response(serializer.errors, 400)
|
|
|
|
data = serializer.validated_data
|
2019-11-18 15:42:42 +01:00
|
|
|
|
2019-12-11 17:33:58 -05:00
|
|
|
if data['render'] == 'svg':
|
2019-12-11 13:39:10 -05:00
|
|
|
# Render and return the elevation as an SVG drawing with the correct content type
|
2020-02-20 13:09:43 -05:00
|
|
|
drawing = rack.get_elevation_svg(
|
|
|
|
face=data['face'],
|
2020-07-29 09:43:13 -04:00
|
|
|
user=request.user,
|
2020-02-20 13:09:43 -05:00
|
|
|
unit_width=data['unit_width'],
|
|
|
|
unit_height=data['unit_height'],
|
|
|
|
legend_width=data['legend_width'],
|
2020-04-03 13:16:35 -04:00
|
|
|
include_images=data['include_images'],
|
|
|
|
base_url=request.build_absolute_uri('/')
|
2020-02-20 13:09:43 -05:00
|
|
|
)
|
2019-12-10 03:18:10 -05:00
|
|
|
return HttpResponse(drawing.tostring(), content_type='image/svg+xml')
|
|
|
|
|
|
|
|
else:
|
2019-12-11 13:39:10 -05:00
|
|
|
# Return a JSON representation of the rack units in the elevation
|
2019-12-11 09:45:08 -05:00
|
|
|
elevation = rack.get_rack_units(
|
|
|
|
face=data['face'],
|
2020-08-03 13:37:32 -04:00
|
|
|
user=request.user,
|
2019-12-11 09:45:08 -05:00
|
|
|
exclude=data['exclude'],
|
|
|
|
expand_devices=data['expand_devices']
|
|
|
|
)
|
2019-11-20 18:27:04 +01:00
|
|
|
|
2020-02-22 08:16:29 -06:00
|
|
|
# Enable filtering rack units by ID
|
|
|
|
q = data['q']
|
|
|
|
if q:
|
|
|
|
elevation = [u for u in elevation if q in str(u['id']) or q in str(u['name'])]
|
|
|
|
|
2019-12-10 03:18:10 -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)
|
2019-11-18 15:42:42 +01:00
|
|
|
|
2019-11-21 16:58:07 +01:00
|
|
|
|
2017-02-16 13:46:58 -05:00
|
|
|
#
|
|
|
|
# Rack reservations
|
|
|
|
#
|
|
|
|
|
2017-11-07 15:36:10 -05:00
|
|
|
class RackReservationViewSet(ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = RackReservation.objects.prefetch_related('rack', 'user', 'tenant')
|
2017-02-16 13:46:58 -05:00
|
|
|
serializer_class = serializers.RackReservationSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.RackReservationFilterSet
|
2017-02-16 13:46:58 -05:00
|
|
|
|
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
#
|
|
|
|
# Manufacturers
|
|
|
|
#
|
|
|
|
|
2021-02-25 15:58:13 -05:00
|
|
|
class ManufacturerViewSet(CustomFieldModelViewSet):
|
2019-04-12 17:07:56 -04:00
|
|
|
queryset = Manufacturer.objects.annotate(
|
2020-12-17 14:47:49 -05:00
|
|
|
devicetype_count=count_related(DeviceType, 'manufacturer'),
|
|
|
|
inventoryitem_count=count_related(InventoryItem, 'manufacturer'),
|
|
|
|
platform_count=count_related(Platform, 'manufacturer')
|
2020-11-25 15:49:18 -05:00
|
|
|
)
|
2016-05-18 16:02:53 -04:00
|
|
|
serializer_class = serializers.ManufacturerSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.ManufacturerFilterSet
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
|
|
|
|
#
|
2017-02-03 17:18:47 -05:00
|
|
|
# Device types
|
2016-03-01 11:23:03 -05:00
|
|
|
#
|
|
|
|
|
2017-11-07 15:36:10 -05:00
|
|
|
class DeviceTypeViewSet(CustomFieldModelViewSet):
|
2020-06-29 13:30:41 -04:00
|
|
|
queryset = DeviceType.objects.prefetch_related('manufacturer', 'tags').annotate(
|
2020-12-17 14:47:49 -05:00
|
|
|
device_count=count_related(Device, 'device_type')
|
2020-11-25 15:49:18 -05:00
|
|
|
)
|
2017-01-27 12:22:29 -05:00
|
|
|
serializer_class = serializers.DeviceTypeSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.DeviceTypeFilterSet
|
2020-12-29 11:41:44 -05:00
|
|
|
brief_prefetch_fields = ['manufacturer']
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
|
|
|
|
#
|
2017-02-03 17:18:47 -05:00
|
|
|
# Device type components
|
|
|
|
#
|
|
|
|
|
2017-11-07 15:36:10 -05:00
|
|
|
class ConsolePortTemplateViewSet(ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = ConsolePortTemplate.objects.prefetch_related('device_type__manufacturer')
|
2017-02-03 17:18:47 -05:00
|
|
|
serializer_class = serializers.ConsolePortTemplateSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.ConsolePortTemplateFilterSet
|
2017-02-03 17:18:47 -05:00
|
|
|
|
|
|
|
|
2017-11-07 15:36:10 -05:00
|
|
|
class ConsoleServerPortTemplateViewSet(ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = ConsoleServerPortTemplate.objects.prefetch_related('device_type__manufacturer')
|
2017-02-03 17:18:47 -05:00
|
|
|
serializer_class = serializers.ConsoleServerPortTemplateSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.ConsoleServerPortTemplateFilterSet
|
2017-02-03 17:18:47 -05:00
|
|
|
|
|
|
|
|
2017-11-07 15:36:10 -05:00
|
|
|
class PowerPortTemplateViewSet(ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = PowerPortTemplate.objects.prefetch_related('device_type__manufacturer')
|
2017-02-03 17:18:47 -05:00
|
|
|
serializer_class = serializers.PowerPortTemplateSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.PowerPortTemplateFilterSet
|
2017-02-03 17:18:47 -05:00
|
|
|
|
|
|
|
|
2017-11-07 15:36:10 -05:00
|
|
|
class PowerOutletTemplateViewSet(ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = PowerOutletTemplate.objects.prefetch_related('device_type__manufacturer')
|
2017-02-03 17:18:47 -05:00
|
|
|
serializer_class = serializers.PowerOutletTemplateSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.PowerOutletTemplateFilterSet
|
2017-02-03 17:18:47 -05:00
|
|
|
|
|
|
|
|
2017-11-07 15:36:10 -05:00
|
|
|
class InterfaceTemplateViewSet(ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = InterfaceTemplate.objects.prefetch_related('device_type__manufacturer')
|
2017-02-03 17:18:47 -05:00
|
|
|
serializer_class = serializers.InterfaceTemplateSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.InterfaceTemplateFilterSet
|
2017-02-03 17:18:47 -05:00
|
|
|
|
|
|
|
|
2018-10-25 12:08:13 -04:00
|
|
|
class FrontPortTemplateViewSet(ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = FrontPortTemplate.objects.prefetch_related('device_type__manufacturer')
|
2018-10-25 12:08:13 -04:00
|
|
|
serializer_class = serializers.FrontPortTemplateSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.FrontPortTemplateFilterSet
|
2018-10-03 14:04:16 -04:00
|
|
|
|
|
|
|
|
2018-10-25 12:08:13 -04:00
|
|
|
class RearPortTemplateViewSet(ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = RearPortTemplate.objects.prefetch_related('device_type__manufacturer')
|
2018-10-25 12:08:13 -04:00
|
|
|
serializer_class = serializers.RearPortTemplateSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.RearPortTemplateFilterSet
|
2018-10-03 14:04:16 -04:00
|
|
|
|
|
|
|
|
2017-11-07 15:36:10 -05:00
|
|
|
class DeviceBayTemplateViewSet(ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = DeviceBayTemplate.objects.prefetch_related('device_type__manufacturer')
|
2017-02-03 17:18:47 -05:00
|
|
|
serializer_class = serializers.DeviceBayTemplateSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.DeviceBayTemplateFilterSet
|
2017-02-03 17:18:47 -05:00
|
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
# Device roles
|
2016-03-01 11:23:03 -05:00
|
|
|
#
|
|
|
|
|
2021-02-25 15:58:13 -05:00
|
|
|
class DeviceRoleViewSet(CustomFieldModelViewSet):
|
2019-04-12 17:07:56 -04:00
|
|
|
queryset = DeviceRole.objects.annotate(
|
2020-12-17 14:47:49 -05:00
|
|
|
device_count=count_related(Device, 'device_role'),
|
|
|
|
virtualmachine_count=count_related(VirtualMachine, 'role')
|
2020-11-25 15:49:18 -05:00
|
|
|
)
|
2016-05-18 16:02:53 -04:00
|
|
|
serializer_class = serializers.DeviceRoleSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.DeviceRoleFilterSet
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
# Platforms
|
|
|
|
#
|
|
|
|
|
2021-02-25 15:58:13 -05:00
|
|
|
class PlatformViewSet(CustomFieldModelViewSet):
|
2019-04-12 17:07:56 -04:00
|
|
|
queryset = Platform.objects.annotate(
|
2020-12-17 14:47:49 -05:00
|
|
|
device_count=count_related(Device, 'platform'),
|
|
|
|
virtualmachine_count=count_related(VirtualMachine, 'platform')
|
2020-11-25 15:49:18 -05:00
|
|
|
)
|
2016-05-18 16:02:53 -04:00
|
|
|
serializer_class = serializers.PlatformSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.PlatformFilterSet
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
# Devices
|
|
|
|
#
|
|
|
|
|
2020-12-17 14:03:10 -05:00
|
|
|
class DeviceViewSet(ConfigContextQuerySetMixin, CustomFieldModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = Device.objects.prefetch_related(
|
2021-03-03 14:28:07 -05:00
|
|
|
'device_type__manufacturer', 'device_role', 'tenant', 'platform', 'site', 'location', 'rack', 'parent_bay',
|
2019-08-19 01:53:39 -04:00
|
|
|
'virtual_chassis__master', 'primary_ip4__nat_outside', 'primary_ip6__nat_outside', 'tags',
|
2017-01-24 17:12:16 -05:00
|
|
|
)
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.DeviceFilterSet
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2018-07-03 12:29:20 -04:00
|
|
|
def get_serializer_class(self):
|
|
|
|
"""
|
2019-02-18 21:37:00 -05:00
|
|
|
Select the specific serializer based on the request context.
|
|
|
|
|
|
|
|
If the `brief` query param equates to True, return the NestedDeviceSerializer
|
|
|
|
|
|
|
|
If the `exclude` query param includes `config_context` as a value, return the DeviceSerializer
|
|
|
|
|
|
|
|
Else, return the DeviceWithConfigContextSerializer
|
2018-07-03 12:29:20 -04:00
|
|
|
"""
|
2018-10-04 16:20:01 -04:00
|
|
|
|
|
|
|
request = self.get_serializer_context()['request']
|
|
|
|
if request.query_params.get('brief', False):
|
|
|
|
return serializers.NestedDeviceSerializer
|
|
|
|
|
2019-02-18 21:37:00 -05:00
|
|
|
elif 'config_context' in request.query_params.get('exclude', []):
|
|
|
|
return serializers.DeviceSerializer
|
|
|
|
|
|
|
|
return serializers.DeviceWithConfigContextSerializer
|
2018-06-27 16:02:34 -04:00
|
|
|
|
2020-01-09 16:39:13 +00:00
|
|
|
@swagger_auto_schema(
|
|
|
|
manual_parameters=[
|
|
|
|
Parameter(
|
|
|
|
name='method',
|
|
|
|
in_='query',
|
|
|
|
required=True,
|
|
|
|
type=openapi.TYPE_STRING
|
|
|
|
)
|
|
|
|
],
|
|
|
|
responses={'200': serializers.DeviceNAPALMSerializer}
|
|
|
|
)
|
2018-06-29 11:48:21 -04:00
|
|
|
@action(detail=True, url_path='napalm')
|
2017-07-14 16:07:28 -04:00
|
|
|
def napalm(self, request, pk):
|
2017-07-14 14:42:56 -04:00
|
|
|
"""
|
|
|
|
Execute a NAPALM method on a Device
|
|
|
|
"""
|
2020-06-29 14:41:43 -04:00
|
|
|
device = get_object_or_404(self.queryset, pk=pk)
|
2017-07-14 14:42:56 -04:00
|
|
|
if not device.primary_ip:
|
|
|
|
raise ServiceUnavailable("This device does not have a primary IP address configured.")
|
|
|
|
if device.platform is None:
|
|
|
|
raise ServiceUnavailable("No platform is configured for this device.")
|
|
|
|
if not device.platform.napalm_driver:
|
2020-11-09 16:48:11 -05:00
|
|
|
raise ServiceUnavailable(f"No NAPALM driver is configured for this device's platform: {device.platform}.")
|
2017-07-14 14:42:56 -04:00
|
|
|
|
2020-07-13 08:36:15 -05:00
|
|
|
# Check for primary IP address from NetBox object
|
|
|
|
if device.primary_ip:
|
|
|
|
host = str(device.primary_ip.address.ip)
|
|
|
|
else:
|
|
|
|
# Raise exception for no IP address and no Name if device.name does not exist
|
|
|
|
if not device.name:
|
|
|
|
raise ServiceUnavailable(
|
2020-11-09 16:48:11 -05:00
|
|
|
"This device does not have a primary IP address or device name to lookup configured."
|
|
|
|
)
|
2020-07-13 08:36:15 -05:00
|
|
|
try:
|
|
|
|
# Attempt to complete a DNS name resolution if no primary_ip is set
|
|
|
|
host = socket.gethostbyname(device.name)
|
|
|
|
except socket.gaierror:
|
|
|
|
# Name lookup failure
|
|
|
|
raise ServiceUnavailable(
|
2020-11-09 16:48:11 -05:00
|
|
|
f"Name lookup failure, unable to resolve IP address for {device.name}. Please set Primary IP or "
|
|
|
|
f"setup name resolution.")
|
2020-07-13 08:36:15 -05:00
|
|
|
|
2017-11-08 09:51:37 -05:00
|
|
|
# Check that NAPALM is installed
|
2017-07-14 14:42:56 -04:00
|
|
|
try:
|
|
|
|
import napalm
|
2018-11-14 10:12:35 -05:00
|
|
|
from napalm.base.exceptions import ModuleImportError
|
2020-11-09 16:48:11 -05:00
|
|
|
except ModuleNotFoundError as e:
|
|
|
|
if getattr(e, 'name') == 'napalm':
|
|
|
|
raise ServiceUnavailable("NAPALM is not installed. Please see the documentation for instructions.")
|
|
|
|
raise e
|
2017-11-08 09:51:37 -05:00
|
|
|
|
|
|
|
# Validate the configured driver
|
2017-07-14 14:42:56 -04:00
|
|
|
try:
|
|
|
|
driver = napalm.get_network_driver(device.platform.napalm_driver)
|
|
|
|
except ModuleImportError:
|
|
|
|
raise ServiceUnavailable("NAPALM driver for platform {} not found: {}.".format(
|
|
|
|
device.platform, device.platform.napalm_driver
|
|
|
|
))
|
|
|
|
|
|
|
|
# Verify user permission
|
2020-06-02 09:36:45 -04:00
|
|
|
if not request.user.has_perm('dcim.napalm_read_device'):
|
2017-07-14 14:42:56 -04:00
|
|
|
return HttpResponseForbidden()
|
|
|
|
|
2017-07-14 16:07:28 -04:00
|
|
|
napalm_methods = request.GET.getlist('method')
|
|
|
|
response = OrderedDict([(m, None) for m in napalm_methods])
|
2020-01-08 15:53:48 +00:00
|
|
|
username = settings.NAPALM_USERNAME
|
|
|
|
password = settings.NAPALM_PASSWORD
|
2018-06-29 11:21:00 -04:00
|
|
|
optional_args = settings.NAPALM_ARGS.copy()
|
|
|
|
if device.platform.napalm_args is not None:
|
|
|
|
optional_args.update(device.platform.napalm_args)
|
2020-01-08 15:53:48 +00:00
|
|
|
|
2020-01-08 16:01:18 +00:00
|
|
|
# Update NAPALM parameters according to the request headers
|
2020-01-08 15:53:48 +00:00
|
|
|
for header in request.headers:
|
2020-01-09 16:48:26 +00:00
|
|
|
if header[:9].lower() != 'x-napalm-':
|
2020-01-08 15:53:48 +00:00
|
|
|
continue
|
|
|
|
|
2020-01-09 16:48:26 +00:00
|
|
|
key = header[9:]
|
2020-01-08 15:53:48 +00:00
|
|
|
if key.lower() == 'username':
|
|
|
|
username = request.headers[header]
|
|
|
|
elif key.lower() == 'password':
|
|
|
|
password = request.headers[header]
|
|
|
|
elif key:
|
2020-01-08 16:01:18 +00:00
|
|
|
optional_args[key.lower()] = request.headers[header]
|
2020-01-08 15:53:48 +00:00
|
|
|
|
2020-07-10 16:18:58 -05:00
|
|
|
# Connect to the device
|
2017-07-14 14:42:56 -04:00
|
|
|
d = driver(
|
2020-07-10 15:12:25 -05:00
|
|
|
hostname=host,
|
2020-01-08 15:53:48 +00:00
|
|
|
username=username,
|
|
|
|
password=password,
|
2017-07-26 11:47:59 -04:00
|
|
|
timeout=settings.NAPALM_TIMEOUT,
|
2018-06-29 11:21:00 -04:00
|
|
|
optional_args=optional_args
|
2017-07-14 14:42:56 -04:00
|
|
|
)
|
|
|
|
try:
|
|
|
|
d.open()
|
|
|
|
except Exception as e:
|
2020-07-10 15:12:25 -05:00
|
|
|
raise ServiceUnavailable("Error connecting to the device at {}: {}".format(host, e))
|
2017-07-14 14:42:56 -04:00
|
|
|
|
2018-07-18 14:46:15 -04:00
|
|
|
# Validate and execute each specified NAPALM method
|
|
|
|
for method in napalm_methods:
|
|
|
|
if not hasattr(driver, method):
|
|
|
|
response[method] = {'error': 'Unknown NAPALM method'}
|
|
|
|
continue
|
|
|
|
if not method.startswith('get_'):
|
|
|
|
response[method] = {'error': 'Only get_* NAPALM methods are supported'}
|
|
|
|
continue
|
|
|
|
try:
|
2021-08-31 23:56:18 -07:00
|
|
|
response[method] = decode_dict(getattr(d, method)())
|
2018-07-18 14:46:15 -04:00
|
|
|
except NotImplementedError:
|
2018-11-14 10:12:35 -05:00
|
|
|
response[method] = {'error': 'Method {} not implemented for NAPALM driver {}'.format(method, driver)}
|
|
|
|
except Exception as e:
|
|
|
|
response[method] = {'error': 'Method {} failed: {}'.format(method, e)}
|
2017-07-14 16:07:28 -04:00
|
|
|
d.close()
|
2018-07-18 14:46:15 -04:00
|
|
|
|
2017-07-14 14:42:56 -04:00
|
|
|
return Response(response)
|
|
|
|
|
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
#
|
2017-02-01 14:34:19 -05:00
|
|
|
# Device components
|
2016-03-01 11:23:03 -05:00
|
|
|
#
|
|
|
|
|
2020-10-02 14:54:16 -04:00
|
|
|
class ConsolePortViewSet(PathEndpointMixin, ModelViewSet):
|
2020-10-06 14:30:46 -04:00
|
|
|
queryset = ConsolePort.objects.prefetch_related('device', '_path__destination', 'cable', '_cable_peer', 'tags')
|
2017-01-26 17:58:36 -05:00
|
|
|
serializer_class = serializers.ConsolePortSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.ConsolePortFilterSet
|
2020-12-29 11:41:44 -05:00
|
|
|
brief_prefetch_fields = ['device']
|
2017-01-26 17:58:36 -05:00
|
|
|
|
|
|
|
|
2020-10-02 14:54:16 -04:00
|
|
|
class ConsoleServerPortViewSet(PathEndpointMixin, ModelViewSet):
|
2020-10-06 14:30:46 -04:00
|
|
|
queryset = ConsoleServerPort.objects.prefetch_related(
|
|
|
|
'device', '_path__destination', 'cable', '_cable_peer', 'tags'
|
|
|
|
)
|
2017-01-26 17:58:36 -05:00
|
|
|
serializer_class = serializers.ConsoleServerPortSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.ConsoleServerPortFilterSet
|
2020-12-29 11:41:44 -05:00
|
|
|
brief_prefetch_fields = ['device']
|
2017-01-26 17:58:36 -05:00
|
|
|
|
|
|
|
|
2020-10-02 14:54:16 -04:00
|
|
|
class PowerPortViewSet(PathEndpointMixin, ModelViewSet):
|
2020-10-06 14:30:46 -04:00
|
|
|
queryset = PowerPort.objects.prefetch_related('device', '_path__destination', 'cable', '_cable_peer', 'tags')
|
2017-01-26 17:58:36 -05:00
|
|
|
serializer_class = serializers.PowerPortSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.PowerPortFilterSet
|
2020-12-29 11:41:44 -05:00
|
|
|
brief_prefetch_fields = ['device']
|
2017-01-26 17:58:36 -05:00
|
|
|
|
|
|
|
|
2020-10-02 14:54:16 -04:00
|
|
|
class PowerOutletViewSet(PathEndpointMixin, ModelViewSet):
|
2020-10-06 14:30:46 -04:00
|
|
|
queryset = PowerOutlet.objects.prefetch_related('device', '_path__destination', 'cable', '_cable_peer', 'tags')
|
2017-01-26 17:58:36 -05:00
|
|
|
serializer_class = serializers.PowerOutletSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.PowerOutletFilterSet
|
2020-12-29 11:41:44 -05:00
|
|
|
brief_prefetch_fields = ['device']
|
2017-01-26 17:58:36 -05:00
|
|
|
|
|
|
|
|
2020-10-02 14:54:16 -04:00
|
|
|
class InterfaceViewSet(PathEndpointMixin, ModelViewSet):
|
2020-10-06 14:30:46 -04:00
|
|
|
queryset = Interface.objects.prefetch_related(
|
2021-03-05 13:49:41 -05:00
|
|
|
'device', 'parent', 'lag', '_path__destination', 'cable', '_cable_peer', 'ip_addresses', 'tags'
|
2020-10-06 14:30:46 -04:00
|
|
|
)
|
2017-01-30 15:35:01 -05:00
|
|
|
serializer_class = serializers.InterfaceSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.InterfaceFilterSet
|
2020-12-29 11:41:44 -05:00
|
|
|
brief_prefetch_fields = ['device']
|
2017-01-26 17:58:36 -05:00
|
|
|
|
|
|
|
|
2020-10-08 10:32:17 -04:00
|
|
|
class FrontPortViewSet(PassThroughPortMixin, ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = FrontPort.objects.prefetch_related('device__device_type__manufacturer', 'rear_port', 'cable', 'tags')
|
2018-10-25 12:08:13 -04:00
|
|
|
serializer_class = serializers.FrontPortSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.FrontPortFilterSet
|
2020-12-29 11:41:44 -05:00
|
|
|
brief_prefetch_fields = ['device']
|
2018-10-03 14:04:16 -04:00
|
|
|
|
|
|
|
|
2020-10-08 10:32:17 -04:00
|
|
|
class RearPortViewSet(PassThroughPortMixin, ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = RearPort.objects.prefetch_related('device__device_type__manufacturer', 'cable', 'tags')
|
2018-10-25 12:08:13 -04:00
|
|
|
serializer_class = serializers.RearPortSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.RearPortFilterSet
|
2020-12-29 11:41:44 -05:00
|
|
|
brief_prefetch_fields = ['device']
|
2018-10-03 14:04:16 -04:00
|
|
|
|
|
|
|
|
2017-11-07 15:36:10 -05:00
|
|
|
class DeviceBayViewSet(ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = DeviceBay.objects.prefetch_related('installed_device').prefetch_related('tags')
|
2017-01-26 17:58:36 -05:00
|
|
|
serializer_class = serializers.DeviceBaySerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.DeviceBayFilterSet
|
2020-12-29 11:41:44 -05:00
|
|
|
brief_prefetch_fields = ['device']
|
2017-01-26 17:58:36 -05:00
|
|
|
|
|
|
|
|
2017-11-07 15:36:10 -05:00
|
|
|
class InventoryItemViewSet(ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = InventoryItem.objects.prefetch_related('device', 'manufacturer').prefetch_related('tags')
|
2017-03-21 12:54:08 -04:00
|
|
|
serializer_class = serializers.InventoryItemSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.InventoryItemFilterSet
|
2020-12-29 11:41:44 -05:00
|
|
|
brief_prefetch_fields = ['device']
|
2017-01-31 12:19:41 -05:00
|
|
|
|
2016-07-05 13:43:19 -04:00
|
|
|
|
2018-10-26 12:25:11 -04:00
|
|
|
#
|
|
|
|
# Cables
|
|
|
|
#
|
|
|
|
|
|
|
|
class CableViewSet(ModelViewSet):
|
2020-06-26 11:09:27 -05:00
|
|
|
metadata_class = ContentTypeMetadata
|
2018-11-01 14:14:31 -04:00
|
|
|
queryset = Cable.objects.prefetch_related(
|
|
|
|
'termination_a', 'termination_b'
|
|
|
|
)
|
2018-10-26 12:25:11 -04:00
|
|
|
serializer_class = serializers.CableSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.CableFilterSet
|
2018-10-26 12:25:11 -04:00
|
|
|
|
|
|
|
|
2017-11-17 16:47:26 -05:00
|
|
|
#
|
|
|
|
# Virtual chassis
|
|
|
|
#
|
|
|
|
|
|
|
|
class VirtualChassisViewSet(ModelViewSet):
|
2019-04-12 17:07:56 -04:00
|
|
|
queryset = VirtualChassis.objects.prefetch_related('tags').annotate(
|
2020-12-17 14:47:49 -05:00
|
|
|
member_count=count_related(Device, 'virtual_chassis')
|
2020-11-25 15:49:18 -05:00
|
|
|
)
|
2017-11-17 16:47:26 -05:00
|
|
|
serializer_class = serializers.VirtualChassisSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.VirtualChassisFilterSet
|
2020-12-29 11:41:44 -05:00
|
|
|
brief_prefetch_fields = ['master']
|
2017-11-17 16:47:26 -05:00
|
|
|
|
|
|
|
|
2019-03-12 11:36:29 -04:00
|
|
|
#
|
|
|
|
# Power panels
|
|
|
|
#
|
|
|
|
|
|
|
|
class PowerPanelViewSet(ModelViewSet):
|
2019-08-19 01:53:39 -04:00
|
|
|
queryset = PowerPanel.objects.prefetch_related(
|
2021-03-03 13:30:33 -05:00
|
|
|
'site', 'location'
|
2019-04-12 17:07:56 -04:00
|
|
|
).annotate(
|
2020-12-17 14:47:49 -05:00
|
|
|
powerfeed_count=count_related(PowerFeed, 'power_panel')
|
2020-11-25 15:49:18 -05:00
|
|
|
)
|
2019-03-12 11:36:29 -04:00
|
|
|
serializer_class = serializers.PowerPanelSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.PowerPanelFilterSet
|
2019-03-12 11:36:29 -04:00
|
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
# Power feeds
|
|
|
|
#
|
|
|
|
|
2020-10-08 13:55:29 -04:00
|
|
|
class PowerFeedViewSet(PathEndpointMixin, CustomFieldModelViewSet):
|
2020-10-06 14:30:46 -04:00
|
|
|
queryset = PowerFeed.objects.prefetch_related(
|
|
|
|
'power_panel', 'rack', '_path__destination', 'cable', '_cable_peer', 'tags'
|
|
|
|
)
|
2019-03-12 11:36:29 -04:00
|
|
|
serializer_class = serializers.PowerFeedSerializer
|
2021-04-29 16:38:56 -04:00
|
|
|
filterset_class = filtersets.PowerFeedFilterSet
|
2019-03-12 11:36:29 -04:00
|
|
|
|
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
#
|
|
|
|
# Miscellaneous
|
|
|
|
#
|
|
|
|
|
2017-03-09 12:18:53 -05:00
|
|
|
class ConnectedDeviceViewSet(ViewSet):
|
2016-03-01 11:23:03 -05:00
|
|
|
"""
|
2017-03-09 12:18:53 -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
|
|
|
|
2018-11-06 00:51:55 -05:00
|
|
|
* `peer_device`: The name of the peer device
|
|
|
|
* `peer_interface`: The name of the peer interface
|
2017-03-09 12:18:53 -05:00
|
|
|
"""
|
2017-08-02 11:17:57 -04:00
|
|
|
permission_classes = [IsAuthenticatedOrLoginNotRequired]
|
2018-11-06 11:44:32 -05:00
|
|
|
_device_param = Parameter(
|
|
|
|
name='peer_device',
|
|
|
|
in_='query',
|
|
|
|
description='The name of the peer device',
|
|
|
|
required=True,
|
|
|
|
type=openapi.TYPE_STRING
|
|
|
|
)
|
|
|
|
_interface_param = Parameter(
|
|
|
|
name='peer_interface',
|
|
|
|
in_='query',
|
|
|
|
description='The name of the peer interface',
|
|
|
|
required=True,
|
|
|
|
type=openapi.TYPE_STRING
|
|
|
|
)
|
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"
|
|
|
|
|
2018-03-16 16:48:08 -04:00
|
|
|
@swagger_auto_schema(
|
2018-11-06 11:44:32 -05:00
|
|
|
manual_parameters=[_device_param, _interface_param],
|
|
|
|
responses={'200': serializers.DeviceSerializer}
|
|
|
|
)
|
2017-03-09 12:18:53 -05:00
|
|
|
def list(self, request):
|
2016-08-22 17:15:20 -04:00
|
|
|
|
2018-03-16 16:48:08 -04:00
|
|
|
peer_device_name = request.query_params.get(self._device_param.name)
|
|
|
|
peer_interface_name = request.query_params.get(self._interface_param.name)
|
2018-11-06 11:44:32 -05:00
|
|
|
|
2017-03-09 12:18:53 -05:00
|
|
|
if not peer_device_name or not peer_interface_name:
|
2018-11-06 00:51:55 -05:00
|
|
|
raise MissingFilterException(detail='Request must include "peer_device" and "peer_interface" filters.')
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2017-03-09 12:18:53 -05:00
|
|
|
# Determine local interface from peer interface's connection
|
2020-06-29 14:41:43 -04:00
|
|
|
peer_interface = get_object_or_404(
|
2020-07-23 12:48:03 -04:00
|
|
|
Interface.objects.all(),
|
2020-06-29 14:41:43 -04:00
|
|
|
device__name=peer_device_name,
|
|
|
|
name=peer_interface_name
|
|
|
|
)
|
2020-10-05 09:56:46 -04:00
|
|
|
local_interface = peer_interface.connected_endpoint
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2017-03-09 12:18:53 -05:00
|
|
|
if local_interface is None:
|
|
|
|
return Response()
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2017-03-09 12:18:53 -05:00
|
|
|
return Response(serializers.DeviceSerializer(local_interface.device, context={'request': request}).data)
|