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/ipam/forms.py

1442 lines
40 KiB
Python
Raw Normal View History

2016-03-01 11:23:03 -05:00
from django import forms
from django.core.exceptions import MultipleObjectsReturned
from django.core.validators import MaxValueValidator, MinValueValidator
from taggit.forms import TagField
2016-03-01 11:23:03 -05:00
from dcim.models import Device, Interface, Rack, Region, Site
2020-01-29 13:53:26 -05:00
from extras.forms import (
AddRemoveTagsForm, CustomFieldBulkEditForm, CustomFieldModelCSVForm, CustomFieldModelForm, CustomFieldFilterForm,
)
2019-10-04 12:08:48 -04:00
from tenancy.forms import TenancyFilterForm, TenancyForm
2016-07-27 11:29:20 -04:00
from tenancy.models import Tenant
from utilities.forms import (
add_blank_choice, APISelect, APISelectMultiple, BootstrapMixin, BulkEditNullBooleanSelect, CSVChoiceField,
DatePicker, DynamicModelChoiceField, DynamicModelMultipleChoiceField, ExpandableIPAddressField,
FlexibleModelChoiceField, ReturnURLForm, SlugField, StaticSelect2, StaticSelect2Multiple, TagFilterField,
BOOLEAN_WITH_BLANK_CHOICES,
)
from virtualization.models import VirtualMachine
from .constants import *
2019-11-27 21:46:53 -05:00
from .choices import *
from .models import Aggregate, IPAddress, Prefix, RIR, Role, Service, VLAN, VLANGroup, VRF
2016-03-01 11:23:03 -05:00
PREFIX_MASK_LENGTH_CHOICES = add_blank_choice([
(i, i) for i in range(PREFIX_LENGTH_MIN, PREFIX_LENGTH_MAX + 1)
])
IPADDRESS_MASK_LENGTH_CHOICES = add_blank_choice([
(i, i) for i in range(IPADDRESS_MASK_LENGTH_MIN, IPADDRESS_MASK_LENGTH_MAX + 1)
])
2016-03-01 11:23:03 -05:00
#
# VRFs
#
2020-01-29 10:49:02 -05:00
class VRFForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
model = VRF
2018-11-27 11:57:29 -05:00
fields = [
'name', 'rd', 'enforce_unique', 'description', 'tenant_group', 'tenant', 'tags',
]
labels = {
'rd': "RD",
}
2016-03-01 11:23:03 -05:00
help_texts = {
'rd': "Route distinguisher in any format",
}
2020-01-29 13:53:26 -05:00
class VRFCSVForm(CustomFieldModelCSVForm):
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.',
}
)
2016-03-01 11:23:03 -05:00
class Meta:
model = VRF
fields = VRF.csv_headers
help_texts = {
'name': 'VRF name',
}
2016-03-01 11:23:03 -05:00
2018-07-10 10:00:21 -04:00
class VRFBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=VRF.objects.all(),
widget=forms.MultipleHiddenInput()
)
tenant = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Tenant.objects.all(),
2019-01-10 21:19:13 -05:00
required=False,
widget=APISelect(
api_url="/api/tenancy/tenants/"
)
2018-11-27 11:57:29 -05:00
)
enforce_unique = forms.NullBooleanField(
2018-11-27 11:57:29 -05:00
required=False,
widget=BulkEditNullBooleanSelect(),
label='Enforce unique space'
)
description = forms.CharField(
max_length=100,
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'tenant', 'description',
]
2016-03-01 11:23:03 -05:00
class VRFFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldFilterForm):
model = VRF
2019-05-09 14:32:49 -04:00
field_order = ['q', 'tenant_group', 'tenant']
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
tag = TagFilterField(model)
2020-01-13 20:16:13 +00:00
2016-07-27 11:29:20 -04:00
2016-05-16 13:04:45 -04:00
#
# RIRs
#
class RIRForm(BootstrapMixin, forms.ModelForm):
2016-05-20 15:32:17 -04:00
slug = SlugField()
2016-05-16 13:04:45 -04:00
class Meta:
model = RIR
2018-11-27 11:57:29 -05:00
fields = [
'name', 'slug', 'is_private',
]
class RIRCSVForm(forms.ModelForm):
slug = SlugField()
class Meta:
model = RIR
fields = RIR.csv_headers
help_texts = {
'name': 'RIR name',
}
class RIRFilterForm(BootstrapMixin, forms.Form):
2018-11-27 11:57:29 -05:00
is_private = forms.NullBooleanField(
required=False,
label='Private',
2019-01-10 21:19:13 -05:00
widget=StaticSelect2(
choices=BOOLEAN_WITH_BLANK_CHOICES
2018-11-27 11:57:29 -05:00
)
)
2016-05-16 13:04:45 -04:00
2016-03-01 11:23:03 -05:00
#
# Aggregates
#
2020-01-29 10:49:02 -05:00
class AggregateForm(BootstrapMixin, CustomFieldModelForm):
rir = DynamicModelChoiceField(
queryset=RIR.objects.all(),
widget=APISelect(
api_url="/api/ipam/rirs/"
)
)
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
model = Aggregate
2018-11-27 11:57:29 -05:00
fields = [
'prefix', 'rir', 'date_added', 'description', 'tags',
]
2016-03-01 11:23:03 -05:00
help_texts = {
'prefix': "IPv4 or IPv6 network",
'rir': "Regional Internet Registry responsible for this prefix",
}
2019-01-10 21:19:13 -05:00
widgets = {
2019-12-28 22:55:00 +00:00
'date_added': DatePicker(),
2019-01-10 21:19:13 -05:00
}
2016-03-01 11:23:03 -05:00
2020-01-29 13:53:26 -05:00
class AggregateCSVForm(CustomFieldModelCSVForm):
rir = forms.ModelChoiceField(
queryset=RIR.objects.all(),
to_field_name='name',
help_text='Name of parent RIR',
error_messages={
'invalid_choice': 'RIR not found.',
}
)
2016-03-01 11:23:03 -05:00
class Meta:
model = Aggregate
fields = Aggregate.csv_headers
2016-03-01 11:23:03 -05:00
2018-07-10 10:00:21 -04:00
class AggregateBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=Aggregate.objects.all(),
widget=forms.MultipleHiddenInput()
)
rir = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=RIR.objects.all(),
required=False,
2019-01-10 21:19:13 -05:00
label='RIR',
widget=APISelect(
api_url="/api/ipam/rirs/"
)
2018-11-27 11:57:29 -05:00
)
date_added = forms.DateField(
required=False
)
description = forms.CharField(
max_length=100,
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'date_added', 'description',
]
2019-12-28 22:55:00 +00:00
widgets = {
'date_added': DatePicker(),
}
2016-03-01 11:23:03 -05:00
class AggregateFilterForm(BootstrapMixin, CustomFieldFilterForm):
model = Aggregate
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
family = forms.ChoiceField(
required=False,
choices=add_blank_choice(IPAddressFamilyChoices),
2019-01-10 21:19:13 -05:00
label='Address family',
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
rir = DynamicModelMultipleChoiceField(
2019-01-10 21:19:13 -05:00
queryset=RIR.objects.all(),
to_field_name='slug',
required=False,
2019-01-10 21:19:13 -05:00
label='RIR',
widget=APISelectMultiple(
api_url="/api/ipam/rirs/",
value_field="slug",
)
)
tag = TagFilterField(model)
2020-01-13 20:16:13 +00:00
2016-03-01 11:23:03 -05:00
2016-05-17 15:04:16 -04:00
#
# Roles
#
class RoleForm(BootstrapMixin, forms.ModelForm):
2016-05-20 15:32:17 -04:00
slug = SlugField()
2016-05-17 15:04:16 -04:00
class Meta:
model = Role
2018-11-27 11:57:29 -05:00
fields = [
'name', 'slug', 'weight', 'description',
2018-11-27 11:57:29 -05:00
]
2016-05-17 15:04:16 -04:00
class RoleCSVForm(forms.ModelForm):
slug = SlugField()
class Meta:
model = Role
fields = Role.csv_headers
help_texts = {
'name': 'Role name',
}
2016-03-01 11:23:03 -05:00
#
# Prefixes
#
2020-01-29 10:49:02 -05:00
class PrefixForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
vrf = DynamicModelChoiceField(
queryset=VRF.objects.all(),
required=False,
2020-02-25 14:12:53 -05:00
label='VRF',
widget=APISelect(
api_url="/api/ipam/vrfs/",
)
)
site = DynamicModelChoiceField(
queryset=Site.objects.all(),
required=False,
2019-01-10 21:19:13 -05:00
widget=APISelect(
api_url="/api/dcim/sites/",
filter_for={
'vlan_group': 'site_id',
'vlan': 'site_id',
},
2018-11-27 11:57:29 -05:00
attrs={
'nullable': 'true',
}
)
)
vlan_group = DynamicModelChoiceField(
queryset=VLANGroup.objects.all(),
required=False,
label='VLAN group',
widget=APISelect(
2019-01-10 21:19:13 -05:00
api_url='/api/ipam/vlan-groups/',
filter_for={
'vlan': 'group_id'
},
2018-11-27 11:57:29 -05:00
attrs={
'nullable': 'true',
}
)
)
vlan = DynamicModelChoiceField(
queryset=VLAN.objects.all(),
required=False,
label='VLAN',
widget=APISelect(
2019-01-10 21:19:13 -05:00
api_url='/api/ipam/vlans/',
2018-11-27 11:57:29 -05:00
display_field='display_name'
)
)
role = DynamicModelChoiceField(
queryset=Role.objects.all(),
required=False,
widget=APISelect(
api_url="/api/ipam/roles/"
)
)
tags = TagField(required=False)
2016-03-01 11:23:03 -05:00
class Meta:
model = Prefix
fields = [
'prefix', 'vrf', 'site', 'vlan', 'status', 'role', 'is_pool', 'description', 'tenant_group', 'tenant',
'tags',
]
2019-01-10 21:19:13 -05:00
widgets = {
'status': StaticSelect2(),
}
2016-03-01 11:23:03 -05:00
def __init__(self, *args, **kwargs):
# Initialize helper selectors
instance = kwargs.get('instance')
initial = kwargs.get('initial', {}).copy()
if instance and instance.vlan is not None:
initial['vlan_group'] = instance.vlan.group
kwargs['initial'] = initial
super().__init__(*args, **kwargs)
2016-03-01 11:23:03 -05:00
self.fields['vrf'].empty_label = 'Global'
2020-01-29 13:53:26 -05:00
class PrefixCSVForm(CustomFieldModelCSVForm):
vrf = FlexibleModelChoiceField(
queryset=VRF.objects.all(),
to_field_name='rd',
required=False,
help_text='Route distinguisher of parent VRF (or {ID})',
error_messages={
'invalid_choice': 'VRF not found.',
}
)
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.',
}
)
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
required=False,
to_field_name='name',
help_text='Name of parent site',
error_messages={
'invalid_choice': 'Site not found.',
}
)
2017-06-07 14:19:08 -04:00
vlan_group = forms.CharField(
help_text='Group name of assigned VLAN',
required=False
)
vlan_vid = forms.IntegerField(
help_text='Numeric ID of assigned VLAN',
required=False
)
status = CSVChoiceField(
2019-11-27 21:46:53 -05:00
choices=PrefixStatusChoices,
help_text='Operational status'
)
role = forms.ModelChoiceField(
queryset=Role.objects.all(),
required=False,
to_field_name='name',
help_text='Functional role',
error_messages={
'invalid_choice': 'Invalid role.',
}
)
2016-03-01 11:23:03 -05:00
class Meta:
model = Prefix
fields = Prefix.csv_headers
def clean(self):
super().clean()
2017-06-07 14:19:08 -04:00
site = self.cleaned_data.get('site')
2017-06-07 14:19:08 -04:00
vlan_group = self.cleaned_data.get('vlan_group')
vlan_vid = self.cleaned_data.get('vlan_vid')
2017-06-07 14:19:08 -04:00
# Validate VLAN
if vlan_group and vlan_vid:
try:
2017-06-07 14:19:08 -04:00
self.instance.vlan = VLAN.objects.get(site=site, group__name=vlan_group, vid=vlan_vid)
except VLAN.DoesNotExist:
if site:
2017-06-07 14:19:08 -04:00
raise forms.ValidationError("VLAN {} not found in site {} group {}".format(
vlan_vid, site, vlan_group
))
else:
2017-06-07 14:19:08 -04:00
raise forms.ValidationError("Global VLAN {} not found in group {}".format(vlan_vid, vlan_group))
except MultipleObjectsReturned:
raise forms.ValidationError(
"Multiple VLANs with VID {} found in group {}".format(vlan_vid, vlan_group)
)
2017-06-07 14:19:08 -04:00
elif vlan_vid:
try:
2017-06-07 14:19:08 -04:00
self.instance.vlan = VLAN.objects.get(site=site, group__isnull=True, vid=vlan_vid)
except VLAN.DoesNotExist:
if site:
2017-06-07 14:19:08 -04:00
raise forms.ValidationError("VLAN {} not found in site {}".format(vlan_vid, site))
else:
raise forms.ValidationError("Global VLAN {} not found".format(vlan_vid))
except MultipleObjectsReturned:
raise forms.ValidationError("Multiple VLANs with VID {} found".format(vlan_vid))
2016-03-01 11:23:03 -05:00
2018-07-10 10:00:21 -04:00
class PrefixBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=Prefix.objects.all(),
widget=forms.MultipleHiddenInput()
)
site = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Site.objects.all(),
2019-01-10 21:19:13 -05:00
required=False,
widget=APISelect(
api_url="/api/dcim/sites/"
)
2018-11-27 11:57:29 -05:00
)
vrf = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=VRF.objects.all(),
required=False,
2019-01-10 21:19:13 -05:00
label='VRF',
widget=APISelect(
api_url="/api/ipam/vrfs/"
)
2018-11-27 11:57:29 -05:00
)
prefix_length = forms.IntegerField(
min_value=PREFIX_LENGTH_MIN,
max_value=PREFIX_LENGTH_MAX,
required=False
)
tenant = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Tenant.objects.all(),
2019-01-10 21:19:13 -05:00
required=False,
widget=APISelect(
api_url="/api/tenancy/tenants/"
)
2018-11-27 11:57:29 -05:00
)
status = forms.ChoiceField(
2019-11-27 21:46:53 -05:00
choices=add_blank_choice(PrefixStatusChoices),
2019-01-10 21:19:13 -05:00
required=False,
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
role = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Role.objects.all(),
2019-01-10 21:19:13 -05:00
required=False,
widget=APISelect(
api_url="/api/ipam/roles/"
)
2018-11-27 11:57:29 -05:00
)
is_pool = forms.NullBooleanField(
required=False,
widget=BulkEditNullBooleanSelect(),
label='Is a pool'
)
description = forms.CharField(
max_length=100,
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'site', 'vrf', 'tenant', 'role', 'description',
]
2016-03-01 11:23:03 -05:00
class PrefixFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldFilterForm):
model = Prefix
2019-05-09 14:32:49 -04:00
field_order = [
'q', 'within_include', 'family', 'mask_length', 'vrf_id', 'status', 'region', 'site', 'role', 'tenant_group',
'tenant', 'is_pool', 'expand',
2019-05-09 14:32:49 -04:00
]
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
within_include = forms.CharField(
required=False,
widget=forms.TextInput(
attrs={
'placeholder': 'Prefix',
}
),
label='Search within'
)
family = forms.ChoiceField(
required=False,
choices=add_blank_choice(IPAddressFamilyChoices),
2019-01-10 21:19:13 -05:00
label='Address family',
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
mask_length = forms.ChoiceField(
required=False,
choices=PREFIX_MASK_LENGTH_CHOICES,
2019-01-10 21:19:13 -05:00
label='Mask length',
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
vrf_id = DynamicModelMultipleChoiceField(
2019-01-10 21:19:13 -05:00
queryset=VRF.objects.all(),
required=False,
label='VRF',
2019-01-10 21:19:13 -05:00
widget=APISelectMultiple(
api_url="/api/ipam/vrfs/",
null_option=True,
)
)
2019-01-10 21:19:13 -05:00
status = forms.MultipleChoiceField(
2019-11-27 21:46:53 -05:00
choices=PrefixStatusChoices,
2019-01-10 21:19:13 -05:00
required=False,
widget=StaticSelect2Multiple()
)
region = DynamicModelMultipleChoiceField(
queryset=Region.objects.all(),
to_field_name='slug',
required=False,
widget=APISelectMultiple(
api_url="/api/dcim/regions/",
value_field="slug",
filter_for={
'site': 'region'
}
)
)
site = DynamicModelMultipleChoiceField(
2019-01-10 21:19:13 -05:00
queryset=Site.objects.all(),
to_field_name='slug',
required=False,
2019-01-10 21:19:13 -05:00
widget=APISelectMultiple(
api_url="/api/dcim/sites/",
value_field="slug",
null_option=True,
)
)
role = DynamicModelMultipleChoiceField(
2019-01-10 21:19:13 -05:00
queryset=Role.objects.all(),
to_field_name='slug',
required=False,
2019-01-10 21:19:13 -05:00
widget=APISelectMultiple(
api_url="/api/ipam/roles/",
value_field="slug",
null_option=True,
)
)
is_pool = forms.NullBooleanField(
required=False,
label='Is a pool',
widget=StaticSelect2(
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
2018-11-27 11:57:29 -05:00
expand = forms.BooleanField(
required=False,
label='Expand prefix hierarchy'
)
tag = TagFilterField(model)
2020-01-13 20:16:13 +00:00
2016-03-01 11:23:03 -05:00
#
# IP addresses
#
2020-01-29 10:49:02 -05:00
class IPAddressForm(BootstrapMixin, TenancyForm, ReturnURLForm, CustomFieldModelForm):
interface = forms.ModelChoiceField(
queryset=Interface.objects.all(),
required=False
)
vrf = DynamicModelChoiceField(
queryset=VRF.objects.all(),
required=False,
label='VRF',
widget=APISelect(
api_url="/api/ipam/vrfs/"
)
)
nat_site = DynamicModelChoiceField(
queryset=Site.objects.all(),
required=False,
label='Site',
2019-01-10 21:19:13 -05:00
widget=APISelect(
api_url="/api/dcim/sites/",
filter_for={
'nat_rack': 'site_id',
'nat_device': 'site_id'
2018-11-27 11:57:29 -05:00
}
)
)
nat_rack = DynamicModelChoiceField(
queryset=Rack.objects.all(),
required=False,
label='Rack',
widget=APISelect(
2019-01-10 21:19:13 -05:00
api_url='/api/dcim/racks/',
display_field='display_name',
2019-01-10 21:19:13 -05:00
filter_for={
'nat_device': 'rack_id'
},
2018-11-27 11:57:29 -05:00
attrs={
'nullable': 'true'
}
)
)
nat_device = DynamicModelChoiceField(
queryset=Device.objects.all(),
required=False,
label='Device',
widget=APISelect(
2019-01-10 21:19:13 -05:00
api_url='/api/dcim/devices/',
display_field='display_name',
2019-01-10 21:19:13 -05:00
filter_for={
'nat_inside': 'device_id'
}
)
)
nat_vrf = forms.ModelChoiceField(
queryset=VRF.objects.all(),
required=False,
label='VRF',
widget=APISelect(
api_url="/api/ipam/vrfs/",
filter_for={
'nat_inside': 'vrf_id'
}
)
)
nat_inside = DynamicModelChoiceField(
queryset=IPAddress.objects.all(),
required=False,
label='IP Address',
widget=APISelect(
2019-01-10 21:19:13 -05:00
api_url='/api/ipam/ip-addresses/',
display_field='address'
)
)
2018-11-27 11:57:29 -05:00
primary_for_parent = forms.BooleanField(
required=False,
label='Make this the primary IP for the device/VM'
)
tags = TagField(
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
model = IPAddress
fields = [
2019-04-22 18:10:28 -04:00
'address', 'vrf', 'status', 'role', 'dns_name', 'description', 'interface', 'primary_for_parent',
'nat_site', 'nat_rack', 'nat_inside', 'tenant_group', 'tenant', 'tags',
]
2019-01-10 21:19:13 -05:00
widgets = {
'status': StaticSelect2(),
'role': StaticSelect2(),
}
2016-03-01 11:23:03 -05:00
def __init__(self, *args, **kwargs):
2016-03-01 11:23:03 -05:00
# Initialize helper selectors
instance = kwargs.get('instance')
initial = kwargs.get('initial', {}).copy()
if instance and instance.nat_inside and instance.nat_inside.device is not None:
initial['nat_site'] = instance.nat_inside.device.site
initial['nat_rack'] = instance.nat_inside.device.rack
initial['nat_device'] = instance.nat_inside.device
kwargs['initial'] = initial
2016-03-01 11:23:03 -05:00
super().__init__(*args, **kwargs)
self.fields['vrf'].empty_label = 'Global'
# Limit interface selections to those belonging to the parent device/VM
if self.instance and self.instance.interface:
self.fields['interface'].queryset = Interface.objects.filter(
device=self.instance.interface.device, virtual_machine=self.instance.interface.virtual_machine
)
else:
self.fields['interface'].choices = []
# Initialize primary_for_parent if IP address is already assigned
if self.instance.pk and self.instance.interface is not None:
parent = self.instance.interface.parent
if (
self.instance.address.version == 4 and parent.primary_ip4_id == self.instance.pk or
self.instance.address.version == 6 and parent.primary_ip6_id == self.instance.pk
):
self.initial['primary_for_parent'] = True
def clean(self):
super().clean()
# Primary IP assignment is only available if an interface has been assigned.
if self.cleaned_data.get('primary_for_parent') and not self.cleaned_data.get('interface'):
self.add_error(
'primary_for_parent', "Only IP addresses assigned to an interface can be designated as primary IPs."
)
def save(self, *args, **kwargs):
ipaddress = super().save(*args, **kwargs)
# Assign/clear this IPAddress as the primary for the associated Device/VirtualMachine.
if self.cleaned_data['primary_for_parent']:
parent = self.cleaned_data['interface'].parent
if ipaddress.address.version == 4:
parent.primary_ip4 = ipaddress
else:
parent.primary_ip6 = ipaddress
parent.save()
elif self.cleaned_data['interface']:
parent = self.cleaned_data['interface'].parent
if ipaddress.address.version == 4 and parent.primary_ip4 == ipaddress:
parent.primary_ip4 = None
parent.save()
elif ipaddress.address.version == 6 and parent.primary_ip6 == ipaddress:
parent.primary_ip6 = None
parent.save()
return ipaddress
2016-03-01 11:23:03 -05:00
class IPAddressBulkCreateForm(BootstrapMixin, forms.Form):
2018-11-27 11:57:29 -05:00
pattern = ExpandableIPAddressField(
label='Address pattern'
)
2020-01-29 10:49:02 -05:00
class IPAddressBulkAddForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
vrf = DynamicModelChoiceField(
queryset=VRF.objects.all(),
required=False,
label='VRF',
widget=APISelect(
api_url="/api/ipam/vrfs/"
)
)
class Meta:
model = IPAddress
2018-11-27 11:57:29 -05:00
fields = [
2019-04-22 18:10:28 -04:00
'address', 'vrf', 'status', 'role', 'dns_name', 'description', 'tenant_group', 'tenant',
2018-11-27 11:57:29 -05:00
]
2019-01-10 21:19:13 -05:00
widgets = {
'status': StaticSelect2(),
'role': StaticSelect2(),
}
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.fields['vrf'].empty_label = 'Global'
2020-01-29 13:53:26 -05:00
class IPAddressCSVForm(CustomFieldModelCSVForm):
vrf = FlexibleModelChoiceField(
queryset=VRF.objects.all(),
to_field_name='rd',
required=False,
help_text='Route distinguisher of parent VRF (or {ID})',
error_messages={
'invalid_choice': 'VRF not found.',
}
)
tenant = forms.ModelChoiceField(
queryset=Tenant.objects.all(),
to_field_name='name',
required=False,
help_text='Name of the assigned tenant',
error_messages={
'invalid_choice': 'Tenant not found.',
}
)
status = CSVChoiceField(
2019-11-27 21:54:01 -05:00
choices=IPAddressStatusChoices,
help_text='Operational status'
)
role = CSVChoiceField(
2019-11-27 22:09:16 -05:00
choices=IPAddressRoleChoices,
required=False,
help_text='Functional role'
)
device = FlexibleModelChoiceField(
queryset=Device.objects.all(),
required=False,
to_field_name='name',
help_text='Name or ID of assigned device',
error_messages={
'invalid_choice': 'Device not found.',
}
)
virtual_machine = forms.ModelChoiceField(
queryset=VirtualMachine.objects.all(),
required=False,
to_field_name='name',
help_text='Name of assigned virtual machine',
error_messages={
'invalid_choice': 'Virtual machine not found.',
}
)
interface_name = forms.CharField(
help_text='Name of assigned interface',
required=False
)
is_primary = forms.BooleanField(
help_text='Make this the primary IP for the assigned device',
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
model = IPAddress
fields = IPAddress.csv_headers
2016-03-01 11:23:03 -05:00
def clean(self):
super().clean()
2017-06-07 14:19:08 -04:00
2016-03-01 11:23:03 -05:00
device = self.cleaned_data.get('device')
virtual_machine = self.cleaned_data.get('virtual_machine')
2016-03-01 11:23:03 -05:00
interface_name = self.cleaned_data.get('interface_name')
is_primary = self.cleaned_data.get('is_primary')
# Validate interface
if interface_name and device:
2016-03-01 11:23:03 -05:00
try:
2017-06-07 14:19:08 -04:00
self.instance.interface = Interface.objects.get(device=device, name=interface_name)
2016-03-01 11:23:03 -05:00
except Interface.DoesNotExist:
raise forms.ValidationError("Invalid interface {} for device {}".format(
interface_name, device
))
elif interface_name and virtual_machine:
try:
self.instance.interface = Interface.objects.get(virtual_machine=virtual_machine, name=interface_name)
except Interface.DoesNotExist:
raise forms.ValidationError("Invalid interface {} for virtual machine {}".format(
interface_name, virtual_machine
))
elif interface_name:
raise forms.ValidationError("Interface given ({}) but parent device/virtual machine not specified".format(
interface_name
))
elif device:
raise forms.ValidationError("Device specified ({}) but interface missing".format(device))
elif virtual_machine:
raise forms.ValidationError("Virtual machine specified ({}) but interface missing".format(virtual_machine))
2016-03-01 11:23:03 -05:00
# Validate is_primary
if is_primary and not device and not virtual_machine:
raise forms.ValidationError("No device or virtual machine specified; cannot set as primary IP")
2016-03-01 11:23:03 -05:00
def save(self, *args, **kwargs):
2016-03-01 11:23:03 -05:00
# Set interface
if self.cleaned_data['device'] and self.cleaned_data['interface_name']:
self.instance.interface = Interface.objects.get(
device=self.cleaned_data['device'],
name=self.cleaned_data['interface_name']
)
elif self.cleaned_data['virtual_machine'] and self.cleaned_data['interface_name']:
self.instance.interface = Interface.objects.get(
virtual_machine=self.cleaned_data['virtual_machine'],
name=self.cleaned_data['interface_name']
)
ipaddress = super().save(*args, **kwargs)
# Set as primary for device/VM
2016-03-01 11:23:03 -05:00
if self.cleaned_data['is_primary']:
parent = self.cleaned_data['device'] or self.cleaned_data['virtual_machine']
if self.instance.address.version == 4:
parent.primary_ip4 = ipaddress
elif self.instance.address.version == 6:
parent.primary_ip6 = ipaddress
parent.save()
2016-03-01 11:23:03 -05:00
return ipaddress
2016-03-01 11:23:03 -05:00
2018-07-10 10:00:21 -04:00
class IPAddressBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=IPAddress.objects.all(),
widget=forms.MultipleHiddenInput()
)
vrf = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=VRF.objects.all(),
required=False,
2019-01-10 21:19:13 -05:00
label='VRF',
widget=APISelect(
api_url="/api/ipam/vrfs/"
)
2018-11-27 11:57:29 -05:00
)
mask_length = forms.IntegerField(
min_value=IPADDRESS_MASK_LENGTH_MIN,
max_value=IPADDRESS_MASK_LENGTH_MAX,
required=False
)
tenant = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Tenant.objects.all(),
2019-01-10 21:19:13 -05:00
required=False,
widget=APISelect(
api_url="/api/tenancy/tenants/"
)
2018-11-27 11:57:29 -05:00
)
status = forms.ChoiceField(
2019-11-27 21:54:01 -05:00
choices=add_blank_choice(IPAddressStatusChoices),
2019-01-10 21:19:13 -05:00
required=False,
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
role = forms.ChoiceField(
2019-11-27 22:09:16 -05:00
choices=add_blank_choice(IPAddressRoleChoices),
2019-01-10 21:19:13 -05:00
required=False,
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
2019-04-22 18:10:28 -04:00
dns_name = forms.CharField(
max_length=255,
required=False
)
2018-11-27 11:57:29 -05:00
description = forms.CharField(
2019-04-22 18:10:28 -04:00
max_length=100,
required=False
2018-11-27 11:57:29 -05:00
)
2016-03-01 11:23:03 -05:00
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
2019-04-22 18:10:28 -04:00
'vrf', 'role', 'tenant', 'dns_name', 'description',
2018-11-27 11:57:29 -05:00
]
2016-03-01 11:23:03 -05:00
class IPAddressAssignForm(BootstrapMixin, forms.Form):
vrf_id = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=VRF.objects.all(),
required=False,
label='VRF',
2019-01-10 21:19:13 -05:00
empty_label='Global',
widget=APISelect(
api_url="/api/ipam/vrfs/"
)
2018-11-27 11:57:29 -05:00
)
2020-01-09 16:26:11 +00:00
q = forms.CharField(
required=False,
2020-01-09 16:26:11 +00:00
label='Search',
2018-11-27 11:57:29 -05:00
)
class IPAddressFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldFilterForm):
model = IPAddress
2019-05-09 14:32:49 -04:00
field_order = [
'q', 'parent', 'family', 'mask_length', 'vrf_id', 'status', 'role', 'assigned_to_interface', 'tenant_group',
'tenant',
2019-05-09 14:32:49 -04:00
]
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
parent = forms.CharField(
required=False,
widget=forms.TextInput(
attrs={
'placeholder': 'Prefix',
}
),
label='Parent Prefix'
)
family = forms.ChoiceField(
required=False,
choices=add_blank_choice(IPAddressFamilyChoices),
2019-01-10 21:19:13 -05:00
label='Address family',
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
mask_length = forms.ChoiceField(
required=False,
choices=IPADDRESS_MASK_LENGTH_CHOICES,
2019-01-10 21:19:13 -05:00
label='Mask length',
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
vrf_id = DynamicModelMultipleChoiceField(
2019-01-10 21:19:13 -05:00
queryset=VRF.objects.all(),
required=False,
label='VRF',
2019-01-10 21:19:13 -05:00
widget=APISelectMultiple(
api_url="/api/ipam/vrfs/",
null_option=True,
)
)
2019-01-10 21:19:13 -05:00
status = forms.MultipleChoiceField(
2019-11-27 21:54:01 -05:00
choices=IPAddressStatusChoices,
2019-01-10 21:19:13 -05:00
required=False,
widget=StaticSelect2Multiple()
)
2019-01-10 21:19:13 -05:00
role = forms.MultipleChoiceField(
2019-11-27 22:09:16 -05:00
choices=IPAddressRoleChoices,
2019-01-10 21:19:13 -05:00
required=False,
widget=StaticSelect2Multiple()
)
assigned_to_interface = forms.NullBooleanField(
required=False,
label='Assigned to an interface',
widget=StaticSelect2(
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
tag = TagFilterField(model)
2020-01-13 20:16:13 +00:00
2016-03-01 11:23:03 -05:00
2016-07-15 13:26:54 -04:00
#
# VLAN groups
#
class VLANGroupForm(BootstrapMixin, forms.ModelForm):
site = DynamicModelChoiceField(
queryset=Site.objects.all(),
required=False,
widget=APISelect(
api_url="/api/dcim/sites/"
)
)
2016-07-15 13:26:54 -04:00
slug = SlugField()
class Meta:
model = VLANGroup
2018-11-27 11:57:29 -05:00
fields = [
'site', 'name', 'slug',
]
2016-07-15 13:26:54 -04:00
class VLANGroupCSVForm(forms.ModelForm):
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
required=False,
to_field_name='name',
help_text='Name of parent site',
error_messages={
'invalid_choice': 'Site not found.',
}
)
slug = SlugField()
class Meta:
model = VLANGroup
fields = VLANGroup.csv_headers
help_texts = {
'name': 'Name of VLAN group',
}
class VLANGroupFilterForm(BootstrapMixin, forms.Form):
region = DynamicModelMultipleChoiceField(
queryset=Region.objects.all(),
to_field_name='slug',
required=False,
widget=APISelectMultiple(
api_url="/api/dcim/regions/",
value_field="slug",
filter_for={
'site': 'region',
}
)
)
site = DynamicModelMultipleChoiceField(
2019-01-10 21:19:13 -05:00
queryset=Site.objects.all(),
2017-02-21 14:53:22 -05:00
to_field_name='slug',
required=False,
2019-01-10 21:19:13 -05:00
widget=APISelectMultiple(
api_url="/api/dcim/sites/",
value_field="slug",
null_option=True,
)
2017-02-21 14:53:22 -05:00
)
2016-07-15 13:26:54 -04:00
2016-03-01 11:23:03 -05:00
#
# VLANs
#
2020-01-29 10:49:02 -05:00
class VLANForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
site = DynamicModelChoiceField(
queryset=Site.objects.all(),
required=False,
2019-01-10 21:19:13 -05:00
widget=APISelect(
api_url="/api/dcim/sites/",
filter_for={
'group': 'site_id'
},
2018-11-27 11:57:29 -05:00
attrs={
'nullable': 'true',
}
)
)
group = DynamicModelChoiceField(
queryset=VLANGroup.objects.all(),
required=False,
widget=APISelect(
2019-01-10 21:19:13 -05:00
api_url='/api/ipam/vlan-groups/',
)
)
role = DynamicModelChoiceField(
queryset=Role.objects.all(),
required=False,
widget=APISelect(
api_url="/api/ipam/roles/"
)
)
tags = TagField(required=False)
2016-03-01 11:23:03 -05:00
class Meta:
model = VLAN
2018-11-27 11:57:29 -05:00
fields = [
'site', 'group', 'vid', 'name', 'status', 'role', 'description', 'tenant_group', 'tenant', 'tags',
]
2016-03-01 11:23:03 -05:00
help_texts = {
2017-02-21 14:53:22 -05:00
'site': "Leave blank if this VLAN spans multiple sites",
2016-07-15 13:26:54 -04:00
'group': "VLAN group (optional)",
2016-03-01 11:23:03 -05:00
'vid': "Configured VLAN ID",
'name': "Configured VLAN name",
'status': "Operational status of this VLAN",
'role': "The primary function of this VLAN",
}
2019-01-10 21:19:13 -05:00
widgets = {
'status': StaticSelect2(),
}
2016-03-01 11:23:03 -05:00
2020-01-29 13:53:26 -05:00
class VLANCSVForm(CustomFieldModelCSVForm):
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
required=False,
to_field_name='name',
help_text='Name of parent site',
error_messages={
'invalid_choice': 'Site not found.',
}
)
group_name = forms.CharField(
2017-06-07 14:19:08 -04:00
help_text='Name of VLAN group',
required=False
)
tenant = forms.ModelChoiceField(
queryset=Tenant.objects.all(),
to_field_name='name',
required=False,
help_text='Name of assigned tenant',
error_messages={
'invalid_choice': 'Tenant not found.',
}
)
status = CSVChoiceField(
2019-11-27 22:15:59 -05:00
choices=VLANStatusChoices,
help_text='Operational status'
)
role = forms.ModelChoiceField(
queryset=Role.objects.all(),
required=False,
to_field_name='name',
help_text='Functional role',
error_messages={
'invalid_choice': 'Invalid role.',
}
)
2016-03-01 11:23:03 -05:00
class Meta:
model = VLAN
fields = VLAN.csv_headers
help_texts = {
'vid': 'Numeric VLAN ID (1-4095)',
'name': 'VLAN name',
}
def clean(self):
super().clean()
2017-06-07 14:19:08 -04:00
site = self.cleaned_data.get('site')
group_name = self.cleaned_data.get('group_name')
2017-06-07 14:19:08 -04:00
# Validate VLAN group
if group_name:
try:
2017-06-07 14:19:08 -04:00
self.instance.group = VLANGroup.objects.get(site=site, name=group_name)
except VLANGroup.DoesNotExist:
2017-06-07 14:19:08 -04:00
if site:
2018-11-27 11:57:29 -05:00
raise forms.ValidationError(
"VLAN group {} not found for site {}".format(group_name, site)
)
2017-06-07 14:19:08 -04:00
else:
2018-11-27 11:57:29 -05:00
raise forms.ValidationError(
"Global VLAN group {} not found".format(group_name)
)
2016-03-01 11:23:03 -05:00
2018-07-10 10:00:21 -04:00
class VLANBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=VLAN.objects.all(),
widget=forms.MultipleHiddenInput()
)
site = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Site.objects.all(),
2019-01-10 21:19:13 -05:00
required=False,
widget=APISelect(
api_url="/api/dcim/sites/"
)
2018-11-27 11:57:29 -05:00
)
group = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=VLANGroup.objects.all(),
2019-01-10 21:19:13 -05:00
required=False,
widget=APISelect(
api_url="/api/ipam/vlan-groups/"
)
2018-11-27 11:57:29 -05:00
)
tenant = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Tenant.objects.all(),
2019-01-10 21:19:13 -05:00
required=False,
widget=APISelect(
api_url="/api/tenancy/tenants/"
)
2018-11-27 11:57:29 -05:00
)
status = forms.ChoiceField(
2019-11-27 22:15:59 -05:00
choices=add_blank_choice(VLANStatusChoices),
2019-01-10 21:19:13 -05:00
required=False,
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
role = DynamicModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Role.objects.all(),
2019-01-10 21:19:13 -05:00
required=False,
widget=APISelect(
api_url="/api/ipam/roles/"
)
2018-11-27 11:57:29 -05:00
)
description = forms.CharField(
max_length=100,
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'site', 'group', 'tenant', 'role', 'description',
]
2016-03-01 11:23:03 -05:00
class VLANFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldFilterForm):
model = VLAN
field_order = ['q', 'region', 'site', 'group_id', 'status', 'role', 'tenant_group', 'tenant']
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
region = DynamicModelMultipleChoiceField(
queryset=Region.objects.all(),
to_field_name='slug',
required=False,
widget=APISelectMultiple(
api_url="/api/dcim/regions/",
value_field="slug",
filter_for={
'site': 'region',
'group_id': 'region'
}
)
)
site = DynamicModelMultipleChoiceField(
2019-01-10 21:19:13 -05:00
queryset=Site.objects.all(),
to_field_name='slug',
required=False,
2019-01-10 21:19:13 -05:00
widget=APISelectMultiple(
api_url="/api/dcim/sites/",
value_field="slug",
null_option=True,
)
)
group_id = DynamicModelMultipleChoiceField(
2019-01-10 21:19:13 -05:00
queryset=VLANGroup.objects.all(),
required=False,
label='VLAN group',
2019-01-10 21:19:13 -05:00
widget=APISelectMultiple(
api_url="/api/ipam/vlan-groups/",
null_option=True,
)
)
2019-01-10 21:19:13 -05:00
status = forms.MultipleChoiceField(
2019-11-27 22:15:59 -05:00
choices=VLANStatusChoices,
2019-01-10 21:19:13 -05:00
required=False,
widget=StaticSelect2Multiple()
)
role = DynamicModelMultipleChoiceField(
2019-01-10 21:19:13 -05:00
queryset=Role.objects.all(),
to_field_name='slug',
required=False,
2019-01-10 21:19:13 -05:00
widget=APISelectMultiple(
api_url="/api/ipam/roles/",
value_field="slug",
null_option=True,
)
)
tag = TagFilterField(model)
2020-01-13 20:16:13 +00:00
#
# Services
#
2020-01-29 10:49:02 -05:00
class ServiceForm(BootstrapMixin, CustomFieldModelForm):
port = forms.IntegerField(
min_value=SERVICE_PORT_MIN,
max_value=SERVICE_PORT_MAX
)
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
class Meta:
model = Service
2018-11-27 11:57:29 -05:00
fields = [
'name', 'protocol', 'port', 'ipaddresses', 'description', 'tags',
]
help_texts = {
'ipaddresses': "IP address assignment is optional. If no IPs are selected, the service is assumed to be "
"reachable via all IPs assigned to the device.",
}
2019-01-10 21:19:13 -05:00
widgets = {
'protocol': StaticSelect2(),
'ipaddresses': StaticSelect2Multiple(),
}
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
2017-08-31 12:50:35 -04:00
# Limit IP address choices to those assigned to interfaces of the parent device/VM
if self.instance.device:
vc_interface_ids = [i['id'] for i in self.instance.device.vc_interfaces.values('id')]
2017-08-31 12:50:35 -04:00
self.fields['ipaddresses'].queryset = IPAddress.objects.filter(
interface_id__in=vc_interface_ids
2017-08-31 12:50:35 -04:00
)
elif self.instance.virtual_machine:
self.fields['ipaddresses'].queryset = IPAddress.objects.filter(
interface__virtual_machine=self.instance.virtual_machine
)
else:
self.fields['ipaddresses'].choices = []
class ServiceFilterForm(BootstrapMixin, CustomFieldFilterForm):
model = Service
q = forms.CharField(
required=False,
label='Search'
)
protocol = forms.ChoiceField(
2019-11-27 22:27:06 -05:00
choices=add_blank_choice(ServiceProtocolChoices),
2019-01-10 21:19:13 -05:00
required=False,
widget=StaticSelect2Multiple()
)
port = forms.IntegerField(
2019-01-10 21:19:13 -05:00
required=False,
)
tag = TagFilterField(model)
2020-01-13 20:16:13 +00:00
class ServiceCSVForm(CustomFieldModelCSVForm):
device = FlexibleModelChoiceField(
queryset=Device.objects.all(),
required=False,
to_field_name='name',
help_text='Name or ID of device',
error_messages={
'invalid_choice': 'Device not found.',
}
)
virtual_machine = FlexibleModelChoiceField(
queryset=VirtualMachine.objects.all(),
required=False,
to_field_name='name',
help_text='Name or ID of virtual machine',
error_messages={
'invalid_choice': 'Virtual machine not found.',
}
)
protocol = CSVChoiceField(
choices=ServiceProtocolChoices,
help_text='IP protocol'
)
class Meta:
model = Service
fields = Service.csv_headers
help_texts = {
}
class ServiceBulkEditForm(BootstrapMixin, CustomFieldBulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=Service.objects.all(),
widget=forms.MultipleHiddenInput()
)
protocol = forms.ChoiceField(
2019-11-27 22:27:06 -05:00
choices=add_blank_choice(ServiceProtocolChoices),
2019-01-10 21:19:13 -05:00
required=False,
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
port = forms.IntegerField(
validators=[
MinValueValidator(1),
MaxValueValidator(65535),
],
required=False
)
description = forms.CharField(
max_length=100,
required=False
)
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'description',
2018-11-27 11:57:29 -05:00
]