2021-09-28 10:25:07 -04:00
|
|
|
from django import forms
|
|
|
|
from django.contrib.contenttypes.models import ContentType
|
|
|
|
from django.core.exceptions import ValidationError
|
2023-07-31 23:52:38 +07:00
|
|
|
from django.utils.translation import gettext_lazy as _
|
2021-09-28 10:25:07 -04:00
|
|
|
|
|
|
|
from dcim.forms.common import InterfaceCommonForm
|
|
|
|
from dcim.models import Device, DeviceRole, Platform, Rack, Region, Site, SiteGroup
|
2023-08-15 01:13:28 +05:30
|
|
|
from extras.models import ConfigTemplate
|
2022-02-07 09:46:38 -05:00
|
|
|
from ipam.models import IPAddress, VLAN, VLANGroup, VRF
|
2022-01-28 15:48:15 -05:00
|
|
|
from netbox.forms import NetBoxModelForm
|
2021-09-28 10:25:07 -04:00
|
|
|
from tenancy.forms import TenancyForm
|
2023-04-14 10:33:53 -04:00
|
|
|
from utilities.forms import BootstrapMixin, ConfirmationForm
|
|
|
|
from utilities.forms.fields import (
|
|
|
|
CommentField, DynamicModelChoiceField, DynamicModelMultipleChoiceField, JSONField, SlugField,
|
2021-09-28 10:25:07 -04:00
|
|
|
)
|
2023-04-03 12:49:26 -04:00
|
|
|
from utilities.forms.widgets import HTMXSelect
|
2021-09-28 10:25:07 -04:00
|
|
|
from virtualization.models import *
|
|
|
|
|
|
|
|
__all__ = (
|
|
|
|
'ClusterAddDevicesForm',
|
|
|
|
'ClusterForm',
|
|
|
|
'ClusterGroupForm',
|
|
|
|
'ClusterRemoveDevicesForm',
|
|
|
|
'ClusterTypeForm',
|
|
|
|
'VirtualMachineForm',
|
|
|
|
'VMInterfaceForm',
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-01-28 15:48:15 -05:00
|
|
|
class ClusterTypeForm(NetBoxModelForm):
|
2021-09-28 10:25:07 -04:00
|
|
|
slug = SlugField()
|
|
|
|
|
2022-10-19 05:35:23 -07:00
|
|
|
fieldsets = (
|
2023-07-31 23:52:38 +07:00
|
|
|
(_('Cluster Type'), (
|
2022-10-19 05:35:23 -07:00
|
|
|
'name', 'slug', 'description', 'tags',
|
|
|
|
)),
|
|
|
|
)
|
|
|
|
|
2021-09-28 10:25:07 -04:00
|
|
|
class Meta:
|
|
|
|
model = ClusterType
|
2021-10-21 10:51:02 -04:00
|
|
|
fields = (
|
|
|
|
'name', 'slug', 'description', 'tags',
|
|
|
|
)
|
2021-09-28 10:25:07 -04:00
|
|
|
|
|
|
|
|
2022-01-28 15:48:15 -05:00
|
|
|
class ClusterGroupForm(NetBoxModelForm):
|
2021-09-28 10:25:07 -04:00
|
|
|
slug = SlugField()
|
|
|
|
|
2022-10-19 05:35:23 -07:00
|
|
|
fieldsets = (
|
2023-07-31 23:52:38 +07:00
|
|
|
(_('Cluster Group'), (
|
2022-10-19 05:35:23 -07:00
|
|
|
'name', 'slug', 'description', 'tags',
|
|
|
|
)),
|
|
|
|
)
|
|
|
|
|
2021-09-28 10:25:07 -04:00
|
|
|
class Meta:
|
|
|
|
model = ClusterGroup
|
2021-10-21 10:51:02 -04:00
|
|
|
fields = (
|
|
|
|
'name', 'slug', 'description', 'tags',
|
|
|
|
)
|
2021-09-28 10:25:07 -04:00
|
|
|
|
|
|
|
|
2022-01-28 15:48:15 -05:00
|
|
|
class ClusterForm(TenancyForm, NetBoxModelForm):
|
2021-09-28 10:25:07 -04:00
|
|
|
type = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Type'),
|
2021-09-28 10:25:07 -04:00
|
|
|
queryset=ClusterType.objects.all()
|
|
|
|
)
|
|
|
|
group = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Group'),
|
2021-09-28 10:25:07 -04:00
|
|
|
queryset=ClusterGroup.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
site = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Site'),
|
2021-09-28 10:25:07 -04:00
|
|
|
queryset=Site.objects.all(),
|
|
|
|
required=False,
|
2023-03-13 12:44:26 -04:00
|
|
|
selector=True
|
2021-09-28 10:25:07 -04:00
|
|
|
)
|
|
|
|
comments = CommentField()
|
|
|
|
|
2022-01-31 15:52:36 -05:00
|
|
|
fieldsets = (
|
2023-07-31 23:52:38 +07:00
|
|
|
(_('Cluster'), ('name', 'type', 'group', 'site', 'status', 'description', 'tags')),
|
|
|
|
(_('Tenancy'), ('tenant_group', 'tenant')),
|
2022-01-31 15:52:36 -05:00
|
|
|
)
|
|
|
|
|
2021-09-28 10:25:07 -04:00
|
|
|
class Meta:
|
|
|
|
model = Cluster
|
|
|
|
fields = (
|
2023-03-13 12:44:26 -04:00
|
|
|
'name', 'type', 'group', 'status', 'tenant', 'site', 'description', 'comments', 'tags',
|
2021-09-28 10:25:07 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class ClusterAddDevicesForm(BootstrapMixin, forms.Form):
|
|
|
|
region = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Region'),
|
2021-09-28 10:25:07 -04:00
|
|
|
queryset=Region.objects.all(),
|
|
|
|
required=False,
|
|
|
|
null_option='None'
|
|
|
|
)
|
|
|
|
site_group = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Site group'),
|
2021-09-28 10:25:07 -04:00
|
|
|
queryset=SiteGroup.objects.all(),
|
|
|
|
required=False,
|
|
|
|
null_option='None'
|
|
|
|
)
|
|
|
|
site = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Site'),
|
2021-09-28 10:25:07 -04:00
|
|
|
queryset=Site.objects.all(),
|
|
|
|
required=False,
|
|
|
|
query_params={
|
|
|
|
'region_id': '$region',
|
|
|
|
'group_id': '$site_group',
|
|
|
|
}
|
|
|
|
)
|
|
|
|
rack = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Rack'),
|
2021-09-28 10:25:07 -04:00
|
|
|
queryset=Rack.objects.all(),
|
|
|
|
required=False,
|
|
|
|
null_option='None',
|
|
|
|
query_params={
|
|
|
|
'site_id': '$site'
|
|
|
|
}
|
|
|
|
)
|
|
|
|
devices = DynamicModelMultipleChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Devices'),
|
2021-09-28 10:25:07 -04:00
|
|
|
queryset=Device.objects.all(),
|
|
|
|
query_params={
|
|
|
|
'site_id': '$site',
|
|
|
|
'rack_id': '$rack',
|
|
|
|
'cluster_id': 'null',
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
fields = [
|
|
|
|
'region', 'site', 'rack', 'devices',
|
|
|
|
]
|
|
|
|
|
|
|
|
def __init__(self, cluster, *args, **kwargs):
|
|
|
|
|
|
|
|
self.cluster = cluster
|
|
|
|
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
|
|
|
self.fields['devices'].choices = []
|
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
super().clean()
|
|
|
|
|
|
|
|
# If the Cluster is assigned to a Site, all Devices must be assigned to that Site.
|
|
|
|
if self.cluster.site is not None:
|
|
|
|
for device in self.cleaned_data.get('devices', []):
|
|
|
|
if device.site != self.cluster.site:
|
|
|
|
raise ValidationError({
|
2023-07-31 23:52:38 +07:00
|
|
|
'devices': _("{} belongs to a different site ({}) than the cluster ({})").format(
|
2021-09-28 10:25:07 -04:00
|
|
|
device, device.site, self.cluster.site
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
class ClusterRemoveDevicesForm(ConfirmationForm):
|
|
|
|
pk = forms.ModelMultipleChoiceField(
|
|
|
|
queryset=Device.objects.all(),
|
|
|
|
widget=forms.MultipleHiddenInput()
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-01-28 15:48:15 -05:00
|
|
|
class VirtualMachineForm(TenancyForm, NetBoxModelForm):
|
2022-05-26 14:59:49 -04:00
|
|
|
site = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Site'),
|
2022-07-12 12:30:07 -04:00
|
|
|
queryset=Site.objects.all(),
|
|
|
|
required=False
|
2022-05-26 14:59:49 -04:00
|
|
|
)
|
2021-09-28 10:25:07 -04:00
|
|
|
cluster = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Cluster'),
|
2021-09-28 10:25:07 -04:00
|
|
|
queryset=Cluster.objects.all(),
|
2022-07-12 12:30:07 -04:00
|
|
|
required=False,
|
2023-03-13 12:44:26 -04:00
|
|
|
selector=True,
|
2021-09-28 10:25:07 -04:00
|
|
|
query_params={
|
2022-05-26 14:59:49 -04:00
|
|
|
'site_id': '$site',
|
2021-09-28 10:25:07 -04:00
|
|
|
}
|
|
|
|
)
|
2022-05-25 16:01:10 -04:00
|
|
|
device = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Device'),
|
2022-05-25 16:01:10 -04:00
|
|
|
queryset=Device.objects.all(),
|
|
|
|
required=False,
|
|
|
|
query_params={
|
2022-07-15 10:19:56 -04:00
|
|
|
'cluster_id': '$cluster',
|
|
|
|
'site_id': '$site',
|
2022-07-12 12:30:07 -04:00
|
|
|
},
|
2022-11-03 11:58:26 -07:00
|
|
|
help_text=_("Optionally pin this VM to a specific host device within the cluster")
|
2022-05-25 16:01:10 -04:00
|
|
|
)
|
2021-09-28 10:25:07 -04:00
|
|
|
role = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Role'),
|
2021-09-28 10:25:07 -04:00
|
|
|
queryset=DeviceRole.objects.all(),
|
|
|
|
required=False,
|
|
|
|
query_params={
|
|
|
|
"vm_role": "True"
|
|
|
|
}
|
|
|
|
)
|
|
|
|
platform = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Platform'),
|
2021-09-28 10:25:07 -04:00
|
|
|
queryset=Platform.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
local_context_data = JSONField(
|
|
|
|
required=False,
|
|
|
|
label=''
|
|
|
|
)
|
2023-08-15 01:13:28 +05:30
|
|
|
config_template = DynamicModelChoiceField(
|
|
|
|
queryset=ConfigTemplate.objects.all(),
|
|
|
|
required=False,
|
|
|
|
label=_('Config template')
|
|
|
|
)
|
2022-11-04 08:28:09 -04:00
|
|
|
comments = CommentField()
|
2021-09-28 10:25:07 -04:00
|
|
|
|
2022-01-31 15:52:36 -05:00
|
|
|
fieldsets = (
|
2023-07-31 23:52:38 +07:00
|
|
|
(_('Virtual Machine'), ('name', 'role', 'status', 'description', 'tags')),
|
|
|
|
(_('Site/Cluster'), ('site', 'cluster', 'device')),
|
|
|
|
(_('Tenancy'), ('tenant_group', 'tenant')),
|
2023-08-15 01:13:28 +05:30
|
|
|
(_('Management'), ('platform', 'primary_ip4', 'primary_ip6', 'config_template')),
|
2023-07-31 23:52:38 +07:00
|
|
|
(_('Resources'), ('vcpus', 'memory', 'disk')),
|
|
|
|
(_('Config Context'), ('local_context_data',)),
|
2022-01-31 15:52:36 -05:00
|
|
|
)
|
|
|
|
|
2021-09-28 10:25:07 -04:00
|
|
|
class Meta:
|
|
|
|
model = VirtualMachine
|
|
|
|
fields = [
|
2023-03-13 12:44:26 -04:00
|
|
|
'name', 'status', 'site', 'cluster', 'device', 'role', 'tenant_group', 'tenant', 'platform', 'primary_ip4',
|
|
|
|
'primary_ip6', 'vcpus', 'memory', 'disk', 'description', 'comments', 'tags', 'local_context_data',
|
2023-08-15 01:13:28 +05:30
|
|
|
'config_template',
|
2021-09-28 10:25:07 -04:00
|
|
|
]
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
|
|
|
if self.instance.pk:
|
|
|
|
|
|
|
|
# Compile list of choices for primary IPv4 and IPv6 addresses
|
|
|
|
for family in [4, 6]:
|
|
|
|
ip_choices = [(None, '---------')]
|
|
|
|
|
|
|
|
# Gather PKs of all interfaces belonging to this VM
|
|
|
|
interface_ids = self.instance.interfaces.values_list('pk', flat=True)
|
|
|
|
|
|
|
|
# Collect interface IPs
|
|
|
|
interface_ips = IPAddress.objects.filter(
|
|
|
|
address__family=family,
|
|
|
|
assigned_object_type=ContentType.objects.get_for_model(VMInterface),
|
|
|
|
assigned_object_id__in=interface_ids
|
|
|
|
)
|
|
|
|
if interface_ips:
|
|
|
|
ip_list = [(ip.id, f'{ip.address} ({ip.assigned_object})') for ip in interface_ips]
|
|
|
|
ip_choices.append(('Interface IPs', ip_list))
|
|
|
|
# Collect NAT IPs
|
|
|
|
nat_ips = IPAddress.objects.prefetch_related('nat_inside').filter(
|
|
|
|
address__family=family,
|
|
|
|
nat_inside__assigned_object_type=ContentType.objects.get_for_model(VMInterface),
|
|
|
|
nat_inside__assigned_object_id__in=interface_ids
|
|
|
|
)
|
|
|
|
if nat_ips:
|
|
|
|
ip_list = [(ip.id, f'{ip.address} (NAT)') for ip in nat_ips]
|
|
|
|
ip_choices.append(('NAT IPs', ip_list))
|
|
|
|
self.fields['primary_ip{}'.format(family)].choices = ip_choices
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
# An object that doesn't exist yet can't have any IPs assigned to it
|
|
|
|
self.fields['primary_ip4'].choices = []
|
|
|
|
self.fields['primary_ip4'].widget.attrs['readonly'] = True
|
|
|
|
self.fields['primary_ip6'].choices = []
|
|
|
|
self.fields['primary_ip6'].widget.attrs['readonly'] = True
|
|
|
|
|
|
|
|
|
2022-01-28 15:48:15 -05:00
|
|
|
class VMInterfaceForm(InterfaceCommonForm, NetBoxModelForm):
|
2022-09-15 10:10:32 -04:00
|
|
|
virtual_machine = DynamicModelChoiceField(
|
2023-07-31 23:52:38 +07:00
|
|
|
label=_('Virtual machine'),
|
2023-03-13 12:44:26 -04:00
|
|
|
queryset=VirtualMachine.objects.all(),
|
|
|
|
selector=True
|
2022-09-15 10:10:32 -04:00
|
|
|
)
|
2021-09-28 10:25:07 -04:00
|
|
|
parent = DynamicModelChoiceField(
|
|
|
|
queryset=VMInterface.objects.all(),
|
|
|
|
required=False,
|
2022-11-03 11:58:26 -07:00
|
|
|
label=_('Parent interface'),
|
2021-12-27 21:25:47 -05:00
|
|
|
query_params={
|
|
|
|
'virtual_machine_id': '$virtual_machine',
|
|
|
|
}
|
2021-09-28 10:25:07 -04:00
|
|
|
)
|
2021-10-21 16:30:18 -04:00
|
|
|
bridge = DynamicModelChoiceField(
|
|
|
|
queryset=VMInterface.objects.all(),
|
|
|
|
required=False,
|
2022-11-03 11:58:26 -07:00
|
|
|
label=_('Bridged interface'),
|
2021-12-27 21:25:47 -05:00
|
|
|
query_params={
|
|
|
|
'virtual_machine_id': '$virtual_machine',
|
|
|
|
}
|
2021-10-21 16:30:18 -04:00
|
|
|
)
|
2021-09-28 10:25:07 -04:00
|
|
|
vlan_group = DynamicModelChoiceField(
|
|
|
|
queryset=VLANGroup.objects.all(),
|
|
|
|
required=False,
|
2022-11-03 11:58:26 -07:00
|
|
|
label=_('VLAN group')
|
2021-09-28 10:25:07 -04:00
|
|
|
)
|
|
|
|
untagged_vlan = DynamicModelChoiceField(
|
|
|
|
queryset=VLAN.objects.all(),
|
|
|
|
required=False,
|
2022-11-03 11:58:26 -07:00
|
|
|
label=_('Untagged VLAN'),
|
2021-09-28 10:25:07 -04:00
|
|
|
query_params={
|
|
|
|
'group_id': '$vlan_group',
|
2021-12-27 21:25:47 -05:00
|
|
|
'available_on_virtualmachine': '$virtual_machine',
|
2021-09-28 10:25:07 -04:00
|
|
|
}
|
|
|
|
)
|
|
|
|
tagged_vlans = DynamicModelMultipleChoiceField(
|
|
|
|
queryset=VLAN.objects.all(),
|
|
|
|
required=False,
|
2022-11-03 11:58:26 -07:00
|
|
|
label=_('Tagged VLANs'),
|
2021-09-28 10:25:07 -04:00
|
|
|
query_params={
|
|
|
|
'group_id': '$vlan_group',
|
2021-12-27 21:25:47 -05:00
|
|
|
'available_on_virtualmachine': '$virtual_machine',
|
2021-09-28 10:25:07 -04:00
|
|
|
}
|
|
|
|
)
|
2022-02-07 09:46:38 -05:00
|
|
|
vrf = DynamicModelChoiceField(
|
|
|
|
queryset=VRF.objects.all(),
|
|
|
|
required=False,
|
2022-11-03 11:58:26 -07:00
|
|
|
label=_('VRF')
|
2022-02-07 09:46:38 -05:00
|
|
|
)
|
2021-09-28 10:25:07 -04:00
|
|
|
|
2022-08-29 15:10:14 -04:00
|
|
|
fieldsets = (
|
2023-07-31 23:52:38 +07:00
|
|
|
(_('Interface'), ('virtual_machine', 'name', 'description', 'tags')),
|
|
|
|
(_('Addressing'), ('vrf', 'mac_address')),
|
|
|
|
(_('Operation'), ('mtu', 'enabled')),
|
|
|
|
(_('Related Interfaces'), ('parent', 'bridge')),
|
|
|
|
(_('802.1Q Switching'), ('mode', 'vlan_group', 'untagged_vlan', 'tagged_vlans')),
|
2022-08-29 15:10:14 -04:00
|
|
|
)
|
|
|
|
|
2021-09-28 10:25:07 -04:00
|
|
|
class Meta:
|
|
|
|
model = VMInterface
|
|
|
|
fields = [
|
2021-10-21 16:30:18 -04:00
|
|
|
'virtual_machine', 'name', 'parent', 'bridge', 'enabled', 'mac_address', 'mtu', 'description', 'mode',
|
2022-06-13 19:14:29 -04:00
|
|
|
'vlan_group', 'untagged_vlan', 'tagged_vlans', 'vrf', 'tags',
|
2021-09-28 10:25:07 -04:00
|
|
|
]
|
|
|
|
labels = {
|
|
|
|
'mode': '802.1Q Mode',
|
|
|
|
}
|
2023-02-18 15:32:26 -05:00
|
|
|
widgets = {
|
2023-04-03 12:49:26 -04:00
|
|
|
'mode': HTMXSelect(),
|
2023-02-18 15:32:26 -05:00
|
|
|
}
|
2022-09-15 10:10:32 -04:00
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
|
|
|
# Disable reassignment of VirtualMachine when editing an existing instance
|
|
|
|
if self.instance.pk:
|
|
|
|
self.fields['virtual_machine'].disabled = True
|