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

665 lines
18 KiB
Python
Raw Normal View History

from django import forms
from django.core.exceptions import ValidationError
2017-08-16 17:00:17 -04:00
from django.db.models import Count
from mptt.forms import TreeNodeChoiceField
from taggit.forms import TagField
from dcim.constants import IFACE_FF_VIRTUAL, IFACE_MODE_ACCESS, IFACE_MODE_TAGGED_ALL
from dcim.forms import INTERFACE_MODE_HELP_TEXT
from dcim.models import Device, DeviceRole, Interface, Platform, Rack, Region, Site
2018-07-10 10:00:21 -04:00
from extras.forms import AddRemoveTagsForm, CustomFieldBulkEditForm, CustomFieldForm, CustomFieldFilterForm
from ipam.models import IPAddress
from tenancy.forms import TenancyForm
from tenancy.models import Tenant
2017-08-18 14:37:19 -04:00
from utilities.forms import (
AnnotatedMultipleChoiceField, APISelect, APISelectMultiple, BootstrapMixin, BulkEditForm, BulkEditNullBooleanSelect,
ChainedFieldsMixin, ChainedModelChoiceField, ChainedModelMultipleChoiceField, CommentField, ComponentForm,
ConfirmationForm, CSVChoiceField, ExpandableNameField, FilterChoiceField, FilterTreeNodeMultipleChoiceField,
JSONField, SlugField, SmallTextarea, add_blank_choice,
2017-08-18 14:37:19 -04:00
)
from .constants import VM_STATUS_CHOICES
from .models import Cluster, ClusterGroup, ClusterType, VirtualMachine
VIFACE_FF_CHOICES = (
(IFACE_FF_VIRTUAL, 'Virtual'),
)
#
# Cluster types
#
class ClusterTypeForm(BootstrapMixin, forms.ModelForm):
slug = SlugField()
class Meta:
model = ClusterType
2018-11-27 11:57:29 -05:00
fields = [
'name', 'slug',
]
class ClusterTypeCSVForm(forms.ModelForm):
slug = SlugField()
class Meta:
model = ClusterType
fields = ClusterType.csv_headers
help_texts = {
'name': 'Name of cluster type',
}
#
# Cluster groups
#
class ClusterGroupForm(BootstrapMixin, forms.ModelForm):
slug = SlugField()
class Meta:
model = ClusterGroup
2018-11-27 11:57:29 -05:00
fields = [
'name', 'slug',
]
class ClusterGroupCSVForm(forms.ModelForm):
slug = SlugField()
class Meta:
model = ClusterGroup
fields = ClusterGroup.csv_headers
help_texts = {
'name': 'Name of cluster group',
}
#
# Clusters
#
class ClusterForm(BootstrapMixin, CustomFieldForm):
2018-11-27 11:57:29 -05:00
comments = CommentField(
widget=SmallTextarea()
)
tags = TagField(
required=False
)
class Meta:
model = Cluster
2018-11-27 11:57:29 -05:00
fields = [
'name', 'type', 'group', 'site', 'comments', 'tags',
]
class ClusterCSVForm(forms.ModelForm):
type = forms.ModelChoiceField(
queryset=ClusterType.objects.all(),
to_field_name='name',
help_text='Name of cluster type',
error_messages={
'invalid_choice': 'Invalid cluster type name.',
}
)
group = forms.ModelChoiceField(
queryset=ClusterGroup.objects.all(),
to_field_name='name',
required=False,
help_text='Name of cluster group',
error_messages={
'invalid_choice': 'Invalid cluster group name.',
}
)
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
to_field_name='name',
required=False,
help_text='Name of assigned site',
error_messages={
'invalid_choice': 'Invalid site name.',
}
)
class Meta:
2017-09-11 15:42:18 -04:00
model = Cluster
fields = Cluster.csv_headers
2018-07-10 10:00:21 -04:00
class ClusterBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=Cluster.objects.all(),
widget=forms.MultipleHiddenInput()
)
type = forms.ModelChoiceField(
queryset=ClusterType.objects.all(),
required=False
)
group = forms.ModelChoiceField(
queryset=ClusterGroup.objects.all(),
required=False
)
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
required=False
)
comments = CommentField(
widget=SmallTextarea()
)
2017-09-11 16:14:05 -04:00
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'group', 'site', 'comments',
]
2017-09-11 16:14:05 -04:00
2017-08-16 17:00:17 -04:00
class ClusterFilterForm(BootstrapMixin, CustomFieldFilterForm):
model = Cluster
q = forms.CharField(required=False, label='Search')
type = FilterChoiceField(
2018-11-27 11:57:29 -05:00
queryset=ClusterType.objects.annotate(
filter_count=Count('clusters')
),
to_field_name='slug',
required=False,
)
2017-08-16 17:00:17 -04:00
group = FilterChoiceField(
2018-11-27 11:57:29 -05:00
queryset=ClusterGroup.objects.annotate(
filter_count=Count('clusters')
),
2017-08-16 17:00:17 -04:00
to_field_name='slug',
null_label='-- None --',
2017-08-16 17:00:17 -04:00
required=False,
)
site = FilterChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Site.objects.annotate(
filter_count=Count('clusters')
),
2017-08-16 17:00:17 -04:00
to_field_name='slug',
null_label='-- None --',
2017-08-16 17:00:17 -04:00
required=False,
)
class ClusterAddDevicesForm(BootstrapMixin, ChainedFieldsMixin, forms.Form):
region = TreeNodeChoiceField(
queryset=Region.objects.all(),
required=False,
widget=forms.Select(
2018-11-27 11:57:29 -05:00
attrs={
'filter-for': 'site',
'nullable': 'true',
}
)
)
site = ChainedModelChoiceField(
queryset=Site.objects.all(),
chains=(
('region', 'region'),
),
required=False,
widget=APISelect(
api_url='/api/dcim/sites/?region_id={{region}}',
2018-11-27 11:57:29 -05:00
attrs={
'filter-for': 'rack',
}
)
)
rack = ChainedModelChoiceField(
queryset=Rack.objects.all(),
chains=(
('site', 'site'),
),
required=False,
widget=APISelect(
api_url='/api/dcim/racks/?site_id={{site}}',
2018-11-27 11:57:29 -05:00
attrs={
'filter-for': 'devices',
'nullable': 'true',
}
)
)
devices = ChainedModelMultipleChoiceField(
queryset=Device.objects.filter(cluster__isnull=True),
chains=(
('site', 'site'),
('rack', 'rack'),
),
widget=APISelectMultiple(
api_url='/api/dcim/devices/?site_id={{site}}&rack_id={{rack}}',
display_field='display_name',
disabled_indicator='cluster'
)
)
class Meta:
2018-11-27 11:57:29 -05:00
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({
'devices': "{} belongs to a different site ({}) than the cluster ({})".format(
device, device.site, self.cluster.site
)
})
class ClusterRemoveDevicesForm(ConfirmationForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=Device.objects.all(),
widget=forms.MultipleHiddenInput()
)
#
# Virtual Machines
#
class VirtualMachineForm(BootstrapMixin, TenancyForm, CustomFieldForm):
2017-08-16 17:00:17 -04:00
cluster_group = forms.ModelChoiceField(
queryset=ClusterGroup.objects.all(),
required=False,
widget=forms.Select(
2018-11-27 11:57:29 -05:00
attrs={
'filter-for': 'cluster',
'nullable': 'true',
}
2017-08-16 17:00:17 -04:00
)
)
cluster = ChainedModelChoiceField(
queryset=Cluster.objects.all(),
chains=(
('group', 'cluster_group'),
),
widget=APISelect(
api_url='/api/virtualization/clusters/?group_id={{cluster_group}}'
)
)
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
local_context_data = JSONField(
required=False
)
class Meta:
model = VirtualMachine
fields = [
'name', 'status', 'cluster_group', 'cluster', 'role', 'tenant', 'platform', 'primary_ip4', 'primary_ip6',
'vcpus', 'memory', 'disk', 'comments', 'tags', 'local_context_data',
]
2018-09-18 11:52:12 -04:00
help_texts = {
2018-11-27 11:57:29 -05:00
'local_context_data': "Local config context data overwrites all sources contexts in the final rendered "
"config context",
2018-09-18 11:52:12 -04:00
}
2017-08-16 17:00:17 -04:00
def __init__(self, *args, **kwargs):
# Initialize helper selector
instance = kwargs.get('instance')
if instance.pk and instance.cluster is not None:
initial = kwargs.get('initial', {}).copy()
initial['cluster_group'] = instance.cluster.group
kwargs['initial'] = initial
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, '---------')]
# Collect interface IPs
interface_ips = IPAddress.objects.select_related('interface').filter(
family=family, interface__virtual_machine=self.instance
)
if interface_ips:
ip_choices.append(
('Interface IPs', [
(ip.id, '{} ({})'.format(ip.address, ip.interface)) for ip in interface_ips
])
)
# Collect NAT IPs
nat_ips = IPAddress.objects.select_related('nat_inside').filter(
family=family, nat_inside__interface__virtual_machine=self.instance
)
if nat_ips:
ip_choices.append(
('NAT IPs', [
(ip.id, '{} ({})'.format(ip.address, ip.nat_inside.address)) for ip in nat_ips
])
)
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
class VirtualMachineCSVForm(forms.ModelForm):
status = CSVChoiceField(
choices=VM_STATUS_CHOICES,
required=False,
help_text='Operational status of device'
)
cluster = forms.ModelChoiceField(
queryset=Cluster.objects.all(),
to_field_name='name',
help_text='Name of parent cluster',
error_messages={
'invalid_choice': 'Invalid cluster name.',
}
)
role = forms.ModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=DeviceRole.objects.filter(
vm_role=True
),
required=False,
to_field_name='name',
help_text='Name of functional role',
error_messages={
'invalid_choice': 'Invalid role name.'
}
)
2017-09-11 15:42:18 -04:00
tenant = forms.ModelChoiceField(
queryset=Tenant.objects.all(),
required=False,
to_field_name='name',
help_text='Name of assigned tenant',
error_messages={
'invalid_choice': 'Tenant not found.'
}
)
platform = forms.ModelChoiceField(
queryset=Platform.objects.all(),
required=False,
to_field_name='name',
help_text='Name of assigned platform',
error_messages={
'invalid_choice': 'Invalid platform.',
}
)
class Meta:
2017-09-11 15:42:18 -04:00
model = VirtualMachine
fields = VirtualMachine.csv_headers
2018-07-10 10:00:21 -04:00
class VirtualMachineBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=VirtualMachine.objects.all(),
widget=forms.MultipleHiddenInput()
)
status = forms.ChoiceField(
choices=add_blank_choice(VM_STATUS_CHOICES),
required=False,
initial=''
)
cluster = forms.ModelChoiceField(
queryset=Cluster.objects.all(),
required=False
)
role = forms.ModelChoiceField(
queryset=DeviceRole.objects.filter(
vm_role=True
),
required=False
)
tenant = forms.ModelChoiceField(
queryset=Tenant.objects.all(),
required=False
)
platform = forms.ModelChoiceField(
queryset=Platform.objects.all(),
required=False
)
vcpus = forms.IntegerField(
required=False,
label='vCPUs'
)
memory = forms.IntegerField(
required=False,
label='Memory (MB)'
)
disk = forms.IntegerField(
required=False,
label='Disk (GB)'
)
comments = CommentField(
widget=SmallTextarea()
)
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'role', 'tenant', 'platform', 'vcpus', 'memory', 'disk', 'comments',
]
2017-08-16 17:00:17 -04:00
class VirtualMachineFilterForm(BootstrapMixin, CustomFieldFilterForm):
model = VirtualMachine
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
2017-08-16 17:00:17 -04:00
cluster_group = FilterChoiceField(
queryset=ClusterGroup.objects.all(),
to_field_name='slug',
null_label='-- None --'
2017-08-16 17:00:17 -04:00
)
cluster_type = FilterChoiceField(
queryset=ClusterType.objects.all(),
to_field_name='slug',
null_label='-- None --'
)
2017-08-16 17:00:17 -04:00
cluster_id = FilterChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Cluster.objects.annotate(
filter_count=Count('virtual_machines')
),
2017-08-16 17:00:17 -04:00
label='Cluster'
)
region = FilterTreeNodeMultipleChoiceField(
queryset=Region.objects.all(),
to_field_name='slug',
required=False,
)
site = FilterChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Site.objects.annotate(
filter_count=Count('clusters__virtual_machines')
),
to_field_name='slug',
null_label='-- None --'
)
role = FilterChoiceField(
2018-11-27 11:57:29 -05:00
queryset=DeviceRole.objects.filter(
vm_role=True
).annotate(
filter_count=Count('virtual_machines')
),
to_field_name='slug',
null_label='-- None --'
)
status = AnnotatedMultipleChoiceField(
choices=VM_STATUS_CHOICES,
annotate=VirtualMachine.objects.all(),
annotate_field='status',
required=False
)
tenant = FilterChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Tenant.objects.annotate(
filter_count=Count('virtual_machines')
),
to_field_name='slug',
null_label='-- None --'
)
platform = FilterChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Platform.objects.annotate(
filter_count=Count('virtual_machines')
),
to_field_name='slug',
null_label='-- None --'
)
2017-08-18 14:37:19 -04:00
#
# VM interfaces
#
class InterfaceForm(BootstrapMixin, forms.ModelForm):
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
2017-08-18 14:37:19 -04:00
class Meta:
model = Interface
fields = [
'virtual_machine', 'name', 'form_factor', 'enabled', 'mac_address', 'mtu', 'description', 'mode', 'tags',
'untagged_vlan', 'tagged_vlans',
]
2017-08-18 14:37:19 -04:00
widgets = {
'virtual_machine': forms.HiddenInput(),
'form_factor': forms.HiddenInput(),
2017-08-18 14:37:19 -04:00
}
labels = {
'mode': '802.1Q Mode',
}
help_texts = {
'mode': INTERFACE_MODE_HELP_TEXT,
}
def clean(self):
super().clean()
# Validate VLAN assignments
tagged_vlans = self.cleaned_data['tagged_vlans']
# Untagged interfaces cannot be assigned tagged VLANs
if self.cleaned_data['mode'] == IFACE_MODE_ACCESS and tagged_vlans:
raise forms.ValidationError({
'mode': "An access interface cannot have tagged VLANs assigned."
})
# Remove all tagged VLAN assignments from "tagged all" interfaces
elif self.cleaned_data['mode'] == IFACE_MODE_TAGGED_ALL:
self.cleaned_data['tagged_vlans'] = []
2017-08-18 14:37:19 -04:00
class InterfaceCreateForm(ComponentForm):
2018-11-27 11:57:29 -05:00
name_pattern = ExpandableNameField(
label='Name'
)
form_factor = forms.ChoiceField(
choices=VIFACE_FF_CHOICES,
initial=IFACE_FF_VIRTUAL,
widget=forms.HiddenInput()
)
enabled = forms.BooleanField(
required=False
)
mtu = forms.IntegerField(
required=False,
min_value=1,
max_value=32767,
label='MTU'
)
mac_address = forms.CharField(
required=False,
label='MAC Address'
)
description = forms.CharField(
max_length=100,
required=False
)
tags = TagField(
required=False
)
2017-08-18 14:37:19 -04:00
def __init__(self, *args, **kwargs):
# Set interfaces enabled by default
kwargs['initial'] = kwargs.get('initial', {}).copy()
kwargs['initial'].update({'enabled': True})
super().__init__(*args, **kwargs)
2017-08-18 14:37:19 -04:00
class InterfaceBulkEditForm(BootstrapMixin, BulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=Interface.objects.all(),
widget=forms.MultipleHiddenInput()
)
enabled = forms.NullBooleanField(
required=False,
widget=BulkEditNullBooleanSelect()
)
mtu = forms.IntegerField(
required=False,
min_value=1,
max_value=32767,
label='MTU'
)
description = forms.CharField(
max_length=100,
required=False
)
2017-08-18 14:37:19 -04:00
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'mtu', 'description',
]
#
# Bulk VirtualMachine component creation
#
class VirtualMachineBulkAddComponentForm(BootstrapMixin, forms.Form):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=VirtualMachine.objects.all(),
widget=forms.MultipleHiddenInput()
)
name_pattern = ExpandableNameField(
label='Name'
)
class VirtualMachineBulkAddInterfaceForm(VirtualMachineBulkAddComponentForm):
2018-11-27 11:57:29 -05:00
form_factor = forms.ChoiceField(
choices=VIFACE_FF_CHOICES,
initial=IFACE_FF_VIRTUAL,
widget=forms.HiddenInput()
)
enabled = forms.BooleanField(
required=False,
initial=True
)
mtu = forms.IntegerField(
required=False,
min_value=1,
max_value=32767,
label='MTU'
)
description = forms.CharField(
max_length=100,
required=False
)