views.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630
  1. from collections import OrderedDict
  2. from django.conf import settings
  3. from django.db.models import Count, F
  4. from django.http import HttpResponseForbidden
  5. from django.shortcuts import get_object_or_404
  6. from drf_yasg import openapi
  7. from drf_yasg.openapi import Parameter
  8. from drf_yasg.utils import swagger_auto_schema
  9. from rest_framework.decorators import action
  10. from rest_framework.mixins import ListModelMixin
  11. from rest_framework.response import Response
  12. from rest_framework.viewsets import GenericViewSet, ViewSet
  13. from circuits.models import Circuit
  14. from dcim import filters
  15. from dcim.models import (
  16. Cable, ConsolePort, ConsolePortTemplate, ConsoleServerPort, ConsoleServerPortTemplate, Device, DeviceBay,
  17. DeviceBayTemplate, DeviceRole, DeviceType, FrontPort, FrontPortTemplate, Interface, InterfaceTemplate,
  18. Manufacturer, InventoryItem, Platform, PowerFeed, PowerOutlet, PowerOutletTemplate, PowerPanel, PowerPort,
  19. PowerPortTemplate, Rack, RackGroup, RackReservation, RackRole, RearPort, RearPortTemplate, Region, Site,
  20. VirtualChassis,
  21. )
  22. from extras.api.serializers import RenderedGraphSerializer
  23. from extras.api.views import CustomFieldModelViewSet
  24. from extras.models import Graph, GRAPH_TYPE_INTERFACE, GRAPH_TYPE_SITE
  25. from ipam.models import Prefix, VLAN
  26. from utilities.api import (
  27. get_serializer_for_model, IsAuthenticatedOrLoginNotRequired, FieldChoicesViewSet, ModelViewSet, ServiceUnavailable,
  28. )
  29. from utilities.utils import get_subquery
  30. from virtualization.models import VirtualMachine
  31. from . import serializers
  32. from .exceptions import MissingFilterException
  33. #
  34. # Field choices
  35. #
  36. class DCIMFieldChoicesViewSet(FieldChoicesViewSet):
  37. fields = (
  38. (Cable, ['length_unit', 'status', 'termination_a_type', 'termination_b_type', 'type']),
  39. (ConsolePort, ['connection_status']),
  40. (Device, ['face', 'status']),
  41. (DeviceType, ['subdevice_role']),
  42. (FrontPort, ['type']),
  43. (FrontPortTemplate, ['type']),
  44. (Interface, ['type', 'mode']),
  45. (InterfaceTemplate, ['type']),
  46. (PowerOutlet, ['feed_leg']),
  47. (PowerOutletTemplate, ['feed_leg']),
  48. (PowerPort, ['connection_status']),
  49. (Rack, ['outer_unit', 'status', 'type', 'width']),
  50. (RearPort, ['type']),
  51. (RearPortTemplate, ['type']),
  52. (Site, ['status']),
  53. )
  54. # Mixins
  55. class CableTraceMixin(object):
  56. @action(detail=True, url_path='trace')
  57. def trace(self, request, pk):
  58. """
  59. Trace a complete cable path and return each segment as a three-tuple of (termination, cable, termination).
  60. """
  61. obj = get_object_or_404(self.queryset.model, pk=pk)
  62. # Initialize the path array
  63. path = []
  64. for near_end, cable, far_end in obj.trace(follow_circuits=True):
  65. # Serialize each object
  66. serializer_a = get_serializer_for_model(near_end, prefix='Nested')
  67. x = serializer_a(near_end, context={'request': request}).data
  68. if cable is not None:
  69. y = serializers.TracedCableSerializer(cable, context={'request': request}).data
  70. else:
  71. y = None
  72. if far_end is not None:
  73. serializer_b = get_serializer_for_model(far_end, prefix='Nested')
  74. z = serializer_b(far_end, context={'request': request}).data
  75. else:
  76. z = None
  77. path.append((x, y, z))
  78. return Response(path)
  79. #
  80. # Regions
  81. #
  82. class RegionViewSet(ModelViewSet):
  83. queryset = Region.objects.annotate(
  84. site_count=Count('sites')
  85. )
  86. serializer_class = serializers.RegionSerializer
  87. filterset_class = filters.RegionFilter
  88. #
  89. # Sites
  90. #
  91. class SiteViewSet(CustomFieldModelViewSet):
  92. queryset = Site.objects.prefetch_related(
  93. 'region', 'tenant', 'tags'
  94. ).annotate(
  95. device_count=get_subquery(Device, 'site'),
  96. rack_count=get_subquery(Rack, 'site'),
  97. prefix_count=get_subquery(Prefix, 'site'),
  98. vlan_count=get_subquery(VLAN, 'site'),
  99. circuit_count=get_subquery(Circuit, 'terminations__site'),
  100. virtualmachine_count=get_subquery(VirtualMachine, 'cluster__site'),
  101. )
  102. serializer_class = serializers.SiteSerializer
  103. filterset_class = filters.SiteFilter
  104. @action(detail=True)
  105. def graphs(self, request, pk=None):
  106. """
  107. A convenience method for rendering graphs for a particular site.
  108. """
  109. site = get_object_or_404(Site, pk=pk)
  110. queryset = Graph.objects.filter(type=GRAPH_TYPE_SITE)
  111. serializer = RenderedGraphSerializer(queryset, many=True, context={'graphed_object': site})
  112. return Response(serializer.data)
  113. #
  114. # Rack groups
  115. #
  116. class RackGroupViewSet(ModelViewSet):
  117. queryset = RackGroup.objects.prefetch_related('site').annotate(
  118. rack_count=Count('racks')
  119. )
  120. serializer_class = serializers.RackGroupSerializer
  121. filterset_class = filters.RackGroupFilter
  122. #
  123. # Rack roles
  124. #
  125. class RackRoleViewSet(ModelViewSet):
  126. queryset = RackRole.objects.annotate(
  127. rack_count=Count('racks')
  128. )
  129. serializer_class = serializers.RackRoleSerializer
  130. filterset_class = filters.RackRoleFilter
  131. #
  132. # Racks
  133. #
  134. class RackViewSet(CustomFieldModelViewSet):
  135. queryset = Rack.objects.prefetch_related(
  136. 'site', 'group__site', 'role', 'tenant', 'tags'
  137. ).annotate(
  138. device_count=get_subquery(Device, 'rack'),
  139. powerfeed_count=get_subquery(PowerFeed, 'rack')
  140. )
  141. serializer_class = serializers.RackSerializer
  142. filterset_class = filters.RackFilter
  143. @action(detail=True)
  144. def units(self, request, pk=None):
  145. """
  146. List rack units (by rack)
  147. """
  148. rack = get_object_or_404(Rack, pk=pk)
  149. face = request.GET.get('face', 0)
  150. exclude_pk = request.GET.get('exclude', None)
  151. if exclude_pk is not None:
  152. try:
  153. exclude_pk = int(exclude_pk)
  154. except ValueError:
  155. exclude_pk = None
  156. elevation = rack.get_rack_units(face, exclude_pk)
  157. # Enable filtering rack units by ID
  158. q = request.GET.get('q', None)
  159. if q:
  160. elevation = [u for u in elevation if q in str(u['id'])]
  161. page = self.paginate_queryset(elevation)
  162. if page is not None:
  163. rack_units = serializers.RackUnitSerializer(page, many=True, context={'request': request})
  164. return self.get_paginated_response(rack_units.data)
  165. #
  166. # Rack reservations
  167. #
  168. class RackReservationViewSet(ModelViewSet):
  169. queryset = RackReservation.objects.prefetch_related('rack', 'user', 'tenant')
  170. serializer_class = serializers.RackReservationSerializer
  171. filterset_class = filters.RackReservationFilter
  172. # Assign user from request
  173. def perform_create(self, serializer):
  174. serializer.save(user=self.request.user)
  175. #
  176. # Manufacturers
  177. #
  178. class ManufacturerViewSet(ModelViewSet):
  179. queryset = Manufacturer.objects.annotate(
  180. devicetype_count=get_subquery(DeviceType, 'manufacturer'),
  181. inventoryitem_count=get_subquery(InventoryItem, 'manufacturer'),
  182. platform_count=get_subquery(Platform, 'manufacturer')
  183. )
  184. serializer_class = serializers.ManufacturerSerializer
  185. filterset_class = filters.ManufacturerFilter
  186. #
  187. # Device types
  188. #
  189. class DeviceTypeViewSet(CustomFieldModelViewSet):
  190. queryset = DeviceType.objects.prefetch_related('manufacturer').prefetch_related('tags').annotate(
  191. device_count=Count('instances')
  192. )
  193. serializer_class = serializers.DeviceTypeSerializer
  194. filterset_class = filters.DeviceTypeFilter
  195. #
  196. # Device type components
  197. #
  198. class ConsolePortTemplateViewSet(ModelViewSet):
  199. queryset = ConsolePortTemplate.objects.prefetch_related('device_type__manufacturer')
  200. serializer_class = serializers.ConsolePortTemplateSerializer
  201. filterset_class = filters.ConsolePortTemplateFilter
  202. class ConsoleServerPortTemplateViewSet(ModelViewSet):
  203. queryset = ConsoleServerPortTemplate.objects.prefetch_related('device_type__manufacturer')
  204. serializer_class = serializers.ConsoleServerPortTemplateSerializer
  205. filterset_class = filters.ConsoleServerPortTemplateFilter
  206. class PowerPortTemplateViewSet(ModelViewSet):
  207. queryset = PowerPortTemplate.objects.prefetch_related('device_type__manufacturer')
  208. serializer_class = serializers.PowerPortTemplateSerializer
  209. filterset_class = filters.PowerPortTemplateFilter
  210. class PowerOutletTemplateViewSet(ModelViewSet):
  211. queryset = PowerOutletTemplate.objects.prefetch_related('device_type__manufacturer')
  212. serializer_class = serializers.PowerOutletTemplateSerializer
  213. filterset_class = filters.PowerOutletTemplateFilter
  214. class InterfaceTemplateViewSet(ModelViewSet):
  215. queryset = InterfaceTemplate.objects.prefetch_related('device_type__manufacturer')
  216. serializer_class = serializers.InterfaceTemplateSerializer
  217. filterset_class = filters.InterfaceTemplateFilter
  218. class FrontPortTemplateViewSet(ModelViewSet):
  219. queryset = FrontPortTemplate.objects.prefetch_related('device_type__manufacturer')
  220. serializer_class = serializers.FrontPortTemplateSerializer
  221. filterset_class = filters.FrontPortTemplateFilter
  222. class RearPortTemplateViewSet(ModelViewSet):
  223. queryset = RearPortTemplate.objects.prefetch_related('device_type__manufacturer')
  224. serializer_class = serializers.RearPortTemplateSerializer
  225. filterset_class = filters.RearPortTemplateFilter
  226. class DeviceBayTemplateViewSet(ModelViewSet):
  227. queryset = DeviceBayTemplate.objects.prefetch_related('device_type__manufacturer')
  228. serializer_class = serializers.DeviceBayTemplateSerializer
  229. filterset_class = filters.DeviceBayTemplateFilter
  230. #
  231. # Device roles
  232. #
  233. class DeviceRoleViewSet(ModelViewSet):
  234. queryset = DeviceRole.objects.annotate(
  235. device_count=get_subquery(Device, 'device_role'),
  236. virtualmachine_count=get_subquery(VirtualMachine, 'role')
  237. )
  238. serializer_class = serializers.DeviceRoleSerializer
  239. filterset_class = filters.DeviceRoleFilter
  240. #
  241. # Platforms
  242. #
  243. class PlatformViewSet(ModelViewSet):
  244. queryset = Platform.objects.annotate(
  245. device_count=get_subquery(Device, 'platform'),
  246. virtualmachine_count=get_subquery(VirtualMachine, 'platform')
  247. )
  248. serializer_class = serializers.PlatformSerializer
  249. filterset_class = filters.PlatformFilter
  250. #
  251. # Devices
  252. #
  253. class DeviceViewSet(CustomFieldModelViewSet):
  254. queryset = Device.objects.prefetch_related(
  255. 'device_type__manufacturer', 'device_role', 'tenant', 'platform', 'site', 'rack', 'parent_bay',
  256. 'virtual_chassis__master', 'primary_ip4__nat_outside', 'primary_ip6__nat_outside', 'tags',
  257. )
  258. filterset_class = filters.DeviceFilter
  259. def get_serializer_class(self):
  260. """
  261. Select the specific serializer based on the request context.
  262. If the `brief` query param equates to True, return the NestedDeviceSerializer
  263. If the `exclude` query param includes `config_context` as a value, return the DeviceSerializer
  264. Else, return the DeviceWithConfigContextSerializer
  265. """
  266. request = self.get_serializer_context()['request']
  267. if request.query_params.get('brief', False):
  268. return serializers.NestedDeviceSerializer
  269. elif 'config_context' in request.query_params.get('exclude', []):
  270. return serializers.DeviceSerializer
  271. return serializers.DeviceWithConfigContextSerializer
  272. @action(detail=True, url_path='napalm')
  273. def napalm(self, request, pk):
  274. """
  275. Execute a NAPALM method on a Device
  276. """
  277. device = get_object_or_404(Device, pk=pk)
  278. if not device.primary_ip:
  279. raise ServiceUnavailable("This device does not have a primary IP address configured.")
  280. if device.platform is None:
  281. raise ServiceUnavailable("No platform is configured for this device.")
  282. if not device.platform.napalm_driver:
  283. raise ServiceUnavailable("No NAPALM driver is configured for this device's platform ().".format(
  284. device.platform
  285. ))
  286. # Check that NAPALM is installed
  287. try:
  288. import napalm
  289. from napalm.base.exceptions import ModuleImportError
  290. except ImportError:
  291. raise ServiceUnavailable("NAPALM is not installed. Please see the documentation for instructions.")
  292. # Validate the configured driver
  293. try:
  294. driver = napalm.get_network_driver(device.platform.napalm_driver)
  295. except ModuleImportError:
  296. raise ServiceUnavailable("NAPALM driver for platform {} not found: {}.".format(
  297. device.platform, device.platform.napalm_driver
  298. ))
  299. # Verify user permission
  300. if not request.user.has_perm('dcim.napalm_read'):
  301. return HttpResponseForbidden()
  302. # Connect to the device
  303. napalm_methods = request.GET.getlist('method')
  304. response = OrderedDict([(m, None) for m in napalm_methods])
  305. ip_address = str(device.primary_ip.address.ip)
  306. optional_args = settings.NAPALM_ARGS.copy()
  307. if device.platform.napalm_args is not None:
  308. optional_args.update(device.platform.napalm_args)
  309. d = driver(
  310. hostname=ip_address,
  311. username=settings.NAPALM_USERNAME,
  312. password=settings.NAPALM_PASSWORD,
  313. timeout=settings.NAPALM_TIMEOUT,
  314. optional_args=optional_args
  315. )
  316. try:
  317. d.open()
  318. except Exception as e:
  319. raise ServiceUnavailable("Error connecting to the device at {}: {}".format(ip_address, e))
  320. # Validate and execute each specified NAPALM method
  321. for method in napalm_methods:
  322. if not hasattr(driver, method):
  323. response[method] = {'error': 'Unknown NAPALM method'}
  324. continue
  325. if not method.startswith('get_'):
  326. response[method] = {'error': 'Only get_* NAPALM methods are supported'}
  327. continue
  328. try:
  329. response[method] = getattr(d, method)()
  330. except NotImplementedError:
  331. response[method] = {'error': 'Method {} not implemented for NAPALM driver {}'.format(method, driver)}
  332. except Exception as e:
  333. response[method] = {'error': 'Method {} failed: {}'.format(method, e)}
  334. d.close()
  335. return Response(response)
  336. #
  337. # Device components
  338. #
  339. class ConsolePortViewSet(CableTraceMixin, ModelViewSet):
  340. queryset = ConsolePort.objects.prefetch_related('device', 'connected_endpoint__device', 'cable', 'tags')
  341. serializer_class = serializers.ConsolePortSerializer
  342. filterset_class = filters.ConsolePortFilter
  343. class ConsoleServerPortViewSet(CableTraceMixin, ModelViewSet):
  344. queryset = ConsoleServerPort.objects.prefetch_related('device', 'connected_endpoint__device', 'cable', 'tags')
  345. serializer_class = serializers.ConsoleServerPortSerializer
  346. filterset_class = filters.ConsoleServerPortFilter
  347. class PowerPortViewSet(CableTraceMixin, ModelViewSet):
  348. queryset = PowerPort.objects.prefetch_related(
  349. 'device', '_connected_poweroutlet__device', '_connected_powerfeed', 'cable', 'tags'
  350. )
  351. serializer_class = serializers.PowerPortSerializer
  352. filterset_class = filters.PowerPortFilter
  353. class PowerOutletViewSet(CableTraceMixin, ModelViewSet):
  354. queryset = PowerOutlet.objects.prefetch_related('device', 'connected_endpoint__device', 'cable', 'tags')
  355. serializer_class = serializers.PowerOutletSerializer
  356. filterset_class = filters.PowerOutletFilter
  357. class InterfaceViewSet(CableTraceMixin, ModelViewSet):
  358. queryset = Interface.objects.prefetch_related(
  359. 'device', '_connected_interface', '_connected_circuittermination', 'cable', 'ip_addresses', 'tags'
  360. ).filter(
  361. device__isnull=False
  362. )
  363. serializer_class = serializers.InterfaceSerializer
  364. filterset_class = filters.InterfaceFilter
  365. @action(detail=True)
  366. def graphs(self, request, pk=None):
  367. """
  368. A convenience method for rendering graphs for a particular interface.
  369. """
  370. interface = get_object_or_404(Interface, pk=pk)
  371. queryset = Graph.objects.filter(type=GRAPH_TYPE_INTERFACE)
  372. serializer = RenderedGraphSerializer(queryset, many=True, context={'graphed_object': interface})
  373. return Response(serializer.data)
  374. class FrontPortViewSet(ModelViewSet):
  375. queryset = FrontPort.objects.prefetch_related('device__device_type__manufacturer', 'rear_port', 'cable', 'tags')
  376. serializer_class = serializers.FrontPortSerializer
  377. filterset_class = filters.FrontPortFilter
  378. class RearPortViewSet(ModelViewSet):
  379. queryset = RearPort.objects.prefetch_related('device__device_type__manufacturer', 'cable', 'tags')
  380. serializer_class = serializers.RearPortSerializer
  381. filterset_class = filters.RearPortFilter
  382. class DeviceBayViewSet(ModelViewSet):
  383. queryset = DeviceBay.objects.prefetch_related('installed_device').prefetch_related('tags')
  384. serializer_class = serializers.DeviceBaySerializer
  385. filterset_class = filters.DeviceBayFilter
  386. class InventoryItemViewSet(ModelViewSet):
  387. queryset = InventoryItem.objects.prefetch_related('device', 'manufacturer').prefetch_related('tags')
  388. serializer_class = serializers.InventoryItemSerializer
  389. filterset_class = filters.InventoryItemFilter
  390. #
  391. # Connections
  392. #
  393. class ConsoleConnectionViewSet(ListModelMixin, GenericViewSet):
  394. queryset = ConsolePort.objects.prefetch_related(
  395. 'device', 'connected_endpoint__device'
  396. ).filter(
  397. connected_endpoint__isnull=False
  398. )
  399. serializer_class = serializers.ConsolePortSerializer
  400. filterset_class = filters.ConsoleConnectionFilter
  401. class PowerConnectionViewSet(ListModelMixin, GenericViewSet):
  402. queryset = PowerPort.objects.prefetch_related(
  403. 'device', 'connected_endpoint__device'
  404. ).filter(
  405. _connected_poweroutlet__isnull=False
  406. )
  407. serializer_class = serializers.PowerPortSerializer
  408. filterset_class = filters.PowerConnectionFilter
  409. class InterfaceConnectionViewSet(ListModelMixin, GenericViewSet):
  410. queryset = Interface.objects.prefetch_related(
  411. 'device', '_connected_interface__device'
  412. ).filter(
  413. # Avoid duplicate connections by only selecting the lower PK in a connected pair
  414. _connected_interface__isnull=False,
  415. pk__lt=F('_connected_interface')
  416. )
  417. serializer_class = serializers.InterfaceConnectionSerializer
  418. filterset_class = filters.InterfaceConnectionFilter
  419. #
  420. # Cables
  421. #
  422. class CableViewSet(ModelViewSet):
  423. queryset = Cable.objects.prefetch_related(
  424. 'termination_a', 'termination_b'
  425. )
  426. serializer_class = serializers.CableSerializer
  427. filterset_class = filters.CableFilter
  428. #
  429. # Virtual chassis
  430. #
  431. class VirtualChassisViewSet(ModelViewSet):
  432. queryset = VirtualChassis.objects.prefetch_related('tags').annotate(
  433. member_count=Count('members')
  434. )
  435. serializer_class = serializers.VirtualChassisSerializer
  436. filterset_class = filters.VirtualChassisFilter
  437. #
  438. # Power panels
  439. #
  440. class PowerPanelViewSet(ModelViewSet):
  441. queryset = PowerPanel.objects.prefetch_related(
  442. 'site', 'rack_group'
  443. ).annotate(
  444. powerfeed_count=Count('powerfeeds')
  445. )
  446. serializer_class = serializers.PowerPanelSerializer
  447. filterset_class = filters.PowerPanelFilter
  448. #
  449. # Power feeds
  450. #
  451. class PowerFeedViewSet(CustomFieldModelViewSet):
  452. queryset = PowerFeed.objects.prefetch_related('power_panel', 'rack', 'tags')
  453. serializer_class = serializers.PowerFeedSerializer
  454. filterset_class = filters.PowerFeedFilter
  455. #
  456. # Miscellaneous
  457. #
  458. class ConnectedDeviceViewSet(ViewSet):
  459. """
  460. This endpoint allows a user to determine what device (if any) is connected to a given peer device and peer
  461. interface. This is useful in a situation where a device boots with no configuration, but can detect its neighbors
  462. via a protocol such as LLDP. Two query parameters must be included in the request:
  463. * `peer_device`: The name of the peer device
  464. * `peer_interface`: The name of the peer interface
  465. """
  466. permission_classes = [IsAuthenticatedOrLoginNotRequired]
  467. _device_param = Parameter(
  468. name='peer_device',
  469. in_='query',
  470. description='The name of the peer device',
  471. required=True,
  472. type=openapi.TYPE_STRING
  473. )
  474. _interface_param = Parameter(
  475. name='peer_interface',
  476. in_='query',
  477. description='The name of the peer interface',
  478. required=True,
  479. type=openapi.TYPE_STRING
  480. )
  481. def get_view_name(self):
  482. return "Connected Device Locator"
  483. @swagger_auto_schema(
  484. manual_parameters=[_device_param, _interface_param],
  485. responses={'200': serializers.DeviceSerializer}
  486. )
  487. def list(self, request):
  488. peer_device_name = request.query_params.get(self._device_param.name)
  489. peer_interface_name = request.query_params.get(self._interface_param.name)
  490. if not peer_device_name or not peer_interface_name:
  491. raise MissingFilterException(detail='Request must include "peer_device" and "peer_interface" filters.')
  492. # Determine local interface from peer interface's connection
  493. peer_interface = get_object_or_404(Interface, device__name=peer_device_name, name=peer_interface_name)
  494. local_interface = peer_interface._connected_interface
  495. if local_interface is None:
  496. return Response()
  497. return Response(serializers.DeviceSerializer(local_interface.device, context={'request': request}).data)