Skip to content

Commit e13bf48

Browse files
Add /api/virtualization/virtual-machines/{id}/render-config/ endpoint (#14287)
* Add /api/virtualization/virtual-machines/{id}/render-config/ endpoint * Update Docstring "Device" -> "Virtual Machine" Docstring should mention "..this Virtual Machine" instead of "...this Device", thanks @LuPo! * Move config rendering logic to new RenderConfigMixin * Add tests for render-config API endpoint --------- Co-authored-by: Jeremy Stretch <[email protected]>
1 parent e767fec commit e13bf48

File tree

5 files changed

+73
-26
lines changed

5 files changed

+73
-26
lines changed

netbox/dcim/api/views.py

Lines changed: 2 additions & 22 deletions
Original file line numberDiff line numberDiff line change
@@ -3,23 +3,20 @@
33
from drf_spectacular.types import OpenApiTypes
44
from drf_spectacular.utils import extend_schema, OpenApiParameter
55
from rest_framework.decorators import action
6-
from rest_framework.renderers import JSONRenderer
76
from rest_framework.response import Response
87
from rest_framework.routers import APIRootView
9-
from rest_framework.status import HTTP_400_BAD_REQUEST
108
from rest_framework.viewsets import ViewSet
119

1210
from circuits.models import Circuit
1311
from dcim import filtersets
1412
from dcim.constants import CABLE_TRACE_SVG_DEFAULT_WIDTH
1513
from dcim.models import *
1614
from dcim.svg import CableTraceSVG
17-
from extras.api.mixins import ConfigContextQuerySetMixin, ConfigTemplateRenderMixin
15+
from extras.api.mixins import ConfigContextQuerySetMixin, RenderConfigMixin
1816
from ipam.models import Prefix, VLAN
1917
from netbox.api.authentication import IsAuthenticatedOrLoginNotRequired
2018
from netbox.api.metadata import ContentTypeMetadata
2119
from netbox.api.pagination import StripCountAnnotationsPaginator
22-
from netbox.api.renderers import TextRenderer
2320
from netbox.api.viewsets import NetBoxModelViewSet, MPTTLockedMixin
2421
from netbox.api.viewsets.mixins import SequentialBulkCreatesMixin
2522
from netbox.constants import NESTED_SERIALIZER_PREFIX
@@ -390,7 +387,7 @@ class PlatformViewSet(NetBoxModelViewSet):
390387
class DeviceViewSet(
391388
SequentialBulkCreatesMixin,
392389
ConfigContextQuerySetMixin,
393-
ConfigTemplateRenderMixin,
390+
RenderConfigMixin,
394391
NetBoxModelViewSet
395392
):
396393
queryset = Device.objects.prefetch_related(
@@ -420,23 +417,6 @@ def get_serializer_class(self):
420417

421418
return serializers.DeviceWithConfigContextSerializer
422419

423-
@action(detail=True, methods=['post'], url_path='render-config', renderer_classes=[JSONRenderer, TextRenderer])
424-
def render_config(self, request, pk):
425-
"""
426-
Resolve and render the preferred ConfigTemplate for this Device.
427-
"""
428-
device = self.get_object()
429-
configtemplate = device.get_config_template()
430-
if not configtemplate:
431-
return Response({'error': 'No config template found for this device.'}, status=HTTP_400_BAD_REQUEST)
432-
433-
# Compile context data
434-
context_data = device.get_config_context()
435-
context_data.update(request.data)
436-
context_data.update({'device': device})
437-
438-
return self.render_configtemplate(request, configtemplate, context_data)
439-
440420

441421
class VirtualDeviceContextViewSet(NetBoxModelViewSet):
442422
queryset = VirtualDeviceContext.objects.prefetch_related(

netbox/dcim/tests/test_api.py

Lines changed: 17 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66
from dcim.choices import *
77
from dcim.constants import *
88
from dcim.models import *
9+
from extras.models import ConfigTemplate
910
from ipam.models import ASN, RIR, VLAN, VRF
1011
from netbox.api.serializers import GenericObjectSerializer
1112
from utilities.testing import APITestCase, APIViewTestCases, create_test_device
@@ -1265,6 +1266,22 @@ def test_rack_fit(self):
12651266

12661267
self.assertHttpStatus(response, status.HTTP_400_BAD_REQUEST)
12671268

1269+
def test_render_config(self):
1270+
configtemplate = ConfigTemplate.objects.create(
1271+
name='Config Template 1',
1272+
template_code='Config for device {{ device.name }}'
1273+
)
1274+
1275+
device = Device.objects.first()
1276+
device.config_template = configtemplate
1277+
device.save()
1278+
1279+
self.add_permissions('dcim.add_device')
1280+
url = reverse('dcim-api:device-detail', kwargs={'pk': device.pk}) + 'render-config/'
1281+
response = self.client.post(url, {}, format='json', **self.header)
1282+
self.assertHttpStatus(response, status.HTTP_200_OK)
1283+
self.assertEqual(response.data['content'], f'Config for device {device.name}')
1284+
12681285

12691286
class ModuleTest(APIViewTestCases.APIViewTestCase):
12701287
model = Module

netbox/extras/api/mixins.py

Lines changed: 34 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,16 @@
11
from jinja2.exceptions import TemplateError
2+
from rest_framework.decorators import action
3+
from rest_framework.renderers import JSONRenderer
24
from rest_framework.response import Response
5+
from rest_framework.status import HTTP_400_BAD_REQUEST
36

7+
from netbox.api.renderers import TextRenderer
48
from .nested_serializers import NestedConfigTemplateSerializer
59

610
__all__ = (
711
'ConfigContextQuerySetMixin',
12+
'ConfigTemplateRenderMixin',
13+
'RenderConfigMixin',
814
)
915

1016

@@ -31,7 +37,9 @@ def get_queryset(self):
3137

3238

3339
class ConfigTemplateRenderMixin:
34-
40+
"""
41+
Provides a method to return a rendered ConfigTemplate as REST API data.
42+
"""
3543
def render_configtemplate(self, request, configtemplate, context):
3644
try:
3745
output = configtemplate.render(context=context)
@@ -50,3 +58,28 @@ def render_configtemplate(self, request, configtemplate, context):
5058
'configtemplate': template_serializer.data,
5159
'content': output
5260
})
61+
62+
63+
class RenderConfigMixin(ConfigTemplateRenderMixin):
64+
"""
65+
Provides a /render-config/ endpoint for REST API views whose model may have a ConfigTemplate assigned.
66+
"""
67+
@action(detail=True, methods=['post'], url_path='render-config', renderer_classes=[JSONRenderer, TextRenderer])
68+
def render_config(self, request, pk):
69+
"""
70+
Resolve and render the preferred ConfigTemplate for this Device.
71+
"""
72+
instance = self.get_object()
73+
object_type = instance._meta.model_name
74+
configtemplate = instance.get_config_template()
75+
if not configtemplate:
76+
return Response({
77+
'error': f'No config template found for this {object_type}.'
78+
}, status=HTTP_400_BAD_REQUEST)
79+
80+
# Compile context data
81+
context_data = instance.get_config_context()
82+
context_data.update(request.data)
83+
context_data.update({object_type: instance})
84+
85+
return self.render_configtemplate(request, configtemplate, context_data)

netbox/virtualization/api/views.py

Lines changed: 3 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
from rest_framework.routers import APIRootView
22

33
from dcim.models import Device
4-
from extras.api.mixins import ConfigContextQuerySetMixin
4+
from extras.api.mixins import ConfigContextQuerySetMixin, RenderConfigMixin
55
from netbox.api.viewsets import NetBoxModelViewSet
66
from utilities.query_functions import CollateAsChar
77
from utilities.utils import count_related
@@ -53,9 +53,9 @@ class ClusterViewSet(NetBoxModelViewSet):
5353
# Virtual machines
5454
#
5555

56-
class VirtualMachineViewSet(ConfigContextQuerySetMixin, NetBoxModelViewSet):
56+
class VirtualMachineViewSet(ConfigContextQuerySetMixin, RenderConfigMixin, NetBoxModelViewSet):
5757
queryset = VirtualMachine.objects.prefetch_related(
58-
'site', 'cluster', 'device', 'role', 'tenant', 'platform', 'primary_ip4', 'primary_ip6', 'tags'
58+
'site', 'cluster', 'device', 'role', 'tenant', 'platform', 'primary_ip4', 'primary_ip6', 'config_template', 'tags'
5959
)
6060
filterset_class = filtersets.VirtualMachineFilterSet
6161

netbox/virtualization/tests/test_api.py

Lines changed: 17 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33

44
from dcim.choices import InterfaceModeChoices
55
from dcim.models import Site
6+
from extras.models import ConfigTemplate
67
from ipam.models import VLAN, VRF
78
from utilities.testing import APITestCase, APIViewTestCases, create_test_device
89
from virtualization.choices import *
@@ -228,6 +229,22 @@ def test_unique_name_per_cluster_constraint(self):
228229
response = self.client.post(url, data, format='json', **self.header)
229230
self.assertHttpStatus(response, status.HTTP_400_BAD_REQUEST)
230231

232+
def test_render_config(self):
233+
configtemplate = ConfigTemplate.objects.create(
234+
name='Config Template 1',
235+
template_code='Config for virtual machine {{ virtualmachine.name }}'
236+
)
237+
238+
vm = VirtualMachine.objects.first()
239+
vm.config_template = configtemplate
240+
vm.save()
241+
242+
self.add_permissions('virtualization.add_virtualmachine')
243+
url = reverse('virtualization-api:virtualmachine-detail', kwargs={'pk': vm.pk}) + 'render-config/'
244+
response = self.client.post(url, {}, format='json', **self.header)
245+
self.assertHttpStatus(response, status.HTTP_200_OK)
246+
self.assertEqual(response.data['content'], f'Config for virtual machine {vm.name}')
247+
231248

232249
class VMInterfaceTest(APIViewTestCases.APIViewTestCase):
233250
model = VMInterface

0 commit comments

Comments
 (0)