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

4049 lines
113 KiB
Python
Raw Normal View History

import re
2016-03-01 11:23:03 -05:00
from django import forms
from django.contrib.auth.models import User
from django.contrib.contenttypes.models import ContentType
from django.contrib.postgres.forms.array import SimpleArrayField
2018-10-31 17:05:25 -04:00
from django.core.exceptions import ObjectDoesNotExist
2019-01-09 23:33:08 -05:00
from django.db.models import Q
from mptt.forms import TreeNodeChoiceField
from netaddr import EUI
from netaddr.core import AddrFormatError
from taggit.forms import TagField
from timezone_field import TimeZoneFormField
2016-03-01 11:23:03 -05:00
from circuits.models import Circuit, Provider
from extras.forms import (
AddRemoveTagsForm, CustomFieldForm, CustomFieldBulkEditForm, CustomFieldFilterForm, LocalConfigContextFilterForm
)
from ipam.models import IPAddress, VLAN, VLANGroup
2019-10-04 12:08:48 -04:00
from tenancy.forms import TenancyFilterForm, TenancyForm
2019-05-09 14:32:49 -04:00
from tenancy.models import Tenant, TenantGroup
2016-05-18 16:02:53 -04:00
from utilities.forms import (
2019-01-09 23:33:08 -05:00
APISelect, APISelectMultiple, add_blank_choice, ArrayFieldSelectMultiple, BootstrapMixin, BulkEditForm,
BulkEditNullBooleanSelect, ChainedFieldsMixin, ChainedModelChoiceField, ColorSelect, CommentField, ComponentForm,
ConfirmationForm, CSVChoiceField, ExpandableNameField, FilterChoiceField, FlexibleModelChoiceField, JSONField,
SelectWithPK, SmallTextarea, SlugField, StaticSelect2, StaticSelect2Multiple, BOOLEAN_WITH_BLANK_CHOICES,
2016-05-18 16:02:53 -04:00
)
from virtualization.models import Cluster, ClusterGroup
2019-10-30 16:31:04 -04:00
from .choices import *
2018-10-03 14:04:16 -04:00
from .constants import *
2016-05-18 16:02:53 -04:00
from .models import (
Cable, DeviceBay, DeviceBayTemplate, ConsolePort, ConsolePortTemplate, ConsoleServerPort, ConsoleServerPortTemplate,
Device, DeviceRole, DeviceType, FrontPort, FrontPortTemplate, Interface, InterfaceTemplate, Manufacturer,
2019-03-11 22:40:52 -04:00
InventoryItem, Platform, PowerFeed, PowerOutlet, PowerOutletTemplate, PowerPanel, PowerPort, PowerPortTemplate,
Rack, RackGroup, RackReservation, RackRole, RearPort, RearPortTemplate, Region, Site, VirtualChassis,
2016-05-18 16:02:53 -04:00
)
2016-03-01 11:23:03 -05:00
DEVICE_BY_PK_RE = r'{\d+\}'
2016-03-01 11:23:03 -05:00
INTERFACE_MODE_HELP_TEXT = """
Access: One untagged VLAN<br />
Tagged: One untagged VLAN and/or one or more tagged VLANs<br />
Tagged All: Implies all VLANs are available (w/optional untagged VLAN)
"""
2016-03-01 11:23:03 -05:00
def get_device_by_name_or_pk(name):
"""
Attempt to retrieve a device by either its name or primary key ('{pk}').
"""
if re.match(DEVICE_BY_PK_RE, name):
pk = name.strip('{}')
device = Device.objects.get(pk=pk)
else:
device = Device.objects.get(name=name)
return device
class InterfaceCommonForm:
2019-10-04 12:08:48 -04:00
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'] = []
class BulkRenameForm(forms.Form):
"""
An extendable form to be used for renaming device components in bulk.
"""
find = forms.CharField()
replace = forms.CharField()
use_regex = forms.BooleanField(
required=False,
initial=True,
label='Use regular expressions'
)
def clean(self):
# Validate regular expression in "find" field
if self.cleaned_data['use_regex']:
try:
re.compile(self.cleaned_data['find'])
except re.error:
raise forms.ValidationError({
'find': "Invalid regular expression"
})
#
# Fields
#
class MACAddressField(forms.Field):
widget = forms.CharField
default_error_messages = {
'invalid': 'MAC address must be in EUI-48 format',
}
def to_python(self, value):
value = super().to_python(value)
# Validate MAC address format
try:
value = EUI(value.strip())
except AddrFormatError:
raise forms.ValidationError(self.error_messages['invalid'], code='invalid')
return value
2017-02-28 12:11:43 -05:00
#
# Regions
#
class RegionForm(BootstrapMixin, forms.ModelForm):
slug = SlugField()
class Meta:
model = Region
2018-11-27 11:57:29 -05:00
fields = [
'parent', 'name', 'slug',
]
widgets = {
'parent': APISelect(
api_url="/api/dcim/regions/"
)
}
2017-02-28 12:11:43 -05:00
2017-07-18 01:50:26 +03:00
class RegionCSVForm(forms.ModelForm):
parent = forms.ModelChoiceField(
queryset=Region.objects.all(),
required=False,
to_field_name='name',
help_text='Name of parent region',
error_messages={
'invalid_choice': 'Region not found.',
}
)
class Meta:
model = Region
fields = Region.csv_headers
2017-07-18 01:50:26 +03:00
help_texts = {
'name': 'Region name',
'slug': 'URL-friendly slug',
}
class RegionFilterForm(BootstrapMixin, forms.Form):
model = Site
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
2016-03-01 11:23:03 -05:00
#
# Sites
#
class SiteForm(BootstrapMixin, TenancyForm, CustomFieldForm):
2018-11-27 11:57:29 -05:00
region = TreeNodeChoiceField(
queryset=Region.objects.all(),
2019-01-03 23:02:05 -05:00
required=False,
widget=APISelect(
api_url="/api/dcim/regions/"
)
2018-11-27 11:57:29 -05:00
)
2016-05-20 15:32:17 -04:00
slug = SlugField()
2016-03-01 11:23:03 -05:00
comments = CommentField()
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
model = Site
2017-02-28 12:11:43 -05:00
fields = [
'name', 'slug', 'status', 'region', 'tenant_group', 'tenant', 'facility', 'asn', 'time_zone', 'description',
'physical_address', 'shipping_address', 'latitude', 'longitude', 'contact_name', 'contact_phone',
'contact_email', 'comments', 'tags',
2017-02-28 12:11:43 -05:00
]
2016-03-01 11:23:03 -05:00
widgets = {
2018-11-27 11:57:29 -05:00
'physical_address': SmallTextarea(
attrs={
'rows': 3,
}
),
'shipping_address': SmallTextarea(
attrs={
'rows': 3,
}
),
2019-01-03 23:02:05 -05:00
'status': StaticSelect2(),
'time_zone': StaticSelect2(),
2016-03-01 11:23:03 -05:00
}
help_texts = {
'name': "Full name of the site",
'facility': "Data center provider and facility (e.g. Equinix NY7)",
'asn': "BGP autonomous system number",
'time_zone': "Local time zone",
'description': "Short description (will appear in sites list)",
2016-03-01 11:23:03 -05:00
'physical_address': "Physical location of the building (e.g. for GPS)",
'shipping_address': "If different from the physical address",
'latitude': "Latitude in decimal format (xx.yyyyyy)",
'longitude': "Longitude in decimal format (xx.yyyyyy)"
2016-03-01 11:23:03 -05:00
}
class SiteCSVForm(forms.ModelForm):
status = CSVChoiceField(
choices=SITE_STATUS_CHOICES,
required=False,
help_text='Operational status'
)
2017-02-28 12:11:43 -05:00
region = forms.ModelChoiceField(
queryset=Region.objects.all(),
required=False,
to_field_name='name',
help_text='Name of assigned region',
error_messages={
'invalid_choice': 'Region not found.',
2017-02-28 12:11:43 -05: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.',
2017-02-28 12:11:43 -05:00
}
)
2016-03-01 11:23:03 -05:00
class Meta:
model = Site
fields = Site.csv_headers
help_texts = {
'name': 'Site name',
'slug': 'URL-friendly slug',
'asn': '32-bit autonomous system number',
}
2016-03-01 11:23:03 -05:00
2018-07-10 10:00:21 -04:00
class SiteBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
pk = forms.ModelMultipleChoiceField(
queryset=Site.objects.all(),
widget=forms.MultipleHiddenInput
)
status = forms.ChoiceField(
choices=add_blank_choice(SITE_STATUS_CHOICES),
required=False,
2019-01-08 15:35:34 -08:00
initial='',
widget=StaticSelect2()
)
region = TreeNodeChoiceField(
queryset=Region.objects.all(),
2019-01-08 15:35:34 -08:00
required=False,
widget=APISelect(
api_url="/api/dcim/regions/"
)
)
tenant = forms.ModelChoiceField(
queryset=Tenant.objects.all(),
2019-01-08 15:35:34 -08:00
required=False,
widget=APISelect(
api_url="/api/tenancy/tenants",
)
)
asn = forms.IntegerField(
min_value=1,
max_value=4294967295,
required=False,
label='ASN'
)
description = forms.CharField(
max_length=100,
required=False
)
time_zone = TimeZoneFormField(
choices=add_blank_choice(TimeZoneFormField().choices),
2019-01-08 15:35:34 -08:00
required=False,
widget=StaticSelect2()
)
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'region', 'tenant', 'asn', 'description', 'time_zone',
]
2016-07-28 15:30:29 -04:00
class SiteFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldFilterForm):
model = Site
2019-05-09 14:32:49 -04:00
field_order = ['q', 'status', 'region', 'tenant_group', 'tenant']
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
2019-01-09 13:44:39 -05:00
status = forms.MultipleChoiceField(
choices=SITE_STATUS_CHOICES,
2019-01-08 15:35:34 -08:00
required=False,
widget=StaticSelect2Multiple()
)
2019-01-08 15:35:34 -08:00
region = forms.ModelMultipleChoiceField(
queryset=Region.objects.all(),
2017-02-28 12:11:43 -05:00
to_field_name='slug',
required=False,
2019-01-08 15:35:34 -08:00
widget=APISelectMultiple(
api_url="/api/dcim/regions/",
value_field="slug",
)
2017-02-28 12:11:43 -05:00
)
2016-03-30 12:26:37 -04:00
#
# Rack groups
#
class RackGroupForm(BootstrapMixin, forms.ModelForm):
2016-05-20 15:32:17 -04:00
slug = SlugField()
class Meta:
model = RackGroup
2018-11-27 11:57:29 -05:00
fields = [
'site', 'name', 'slug',
]
widgets = {
'site': APISelect(
api_url="/api/dcim/sites/"
)
}
class RackGroupCSVForm(forms.ModelForm):
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
to_field_name='name',
help_text='Name of parent site',
error_messages={
'invalid_choice': 'Site not found.',
}
)
class Meta:
model = RackGroup
fields = RackGroup.csv_headers
help_texts = {
'name': 'Name of rack group',
'slug': 'URL-friendly slug',
}
class RackGroupFilterForm(BootstrapMixin, forms.Form):
2018-11-27 11:57:29 -05:00
site = FilterChoiceField(
2019-01-09 13:44:39 -05:00
queryset=Site.objects.all(),
2019-01-08 15:35:34 -08:00
to_field_name='slug',
widget=APISelectMultiple(
api_url="/api/dcim/sites/",
value_field="slug",
)
2018-11-27 11:57:29 -05:00
)
2016-03-30 12:26:37 -04:00
2016-08-10 11:52:27 -04:00
#
# Rack roles
#
class RackRoleForm(BootstrapMixin, forms.ModelForm):
2016-08-10 11:52:27 -04:00
slug = SlugField()
class Meta:
model = RackRole
2018-11-27 11:57:29 -05:00
fields = [
'name', 'slug', 'color',
]
2016-08-10 11:52:27 -04:00
class RackRoleCSVForm(forms.ModelForm):
slug = SlugField()
class Meta:
model = RackRole
fields = RackRole.csv_headers
help_texts = {
'name': 'Name of rack role',
'color': 'RGB color in hexadecimal (e.g. 00ff00)'
}
2016-03-01 11:23:03 -05:00
#
# Racks
#
class RackForm(BootstrapMixin, TenancyForm, CustomFieldForm):
group = ChainedModelChoiceField(
queryset=RackGroup.objects.all(),
chains=(
('site', 'site'),
),
required=False,
widget=APISelect(
api_url='/api/dcim/rack-groups/',
)
)
2016-03-01 11:23:03 -05:00
comments = CommentField()
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
model = Rack
fields = [
2018-11-02 09:17:51 -04:00
'site', 'group', 'name', 'facility_id', 'tenant_group', 'tenant', 'status', 'role', 'serial', 'asset_tag',
'type', 'width', 'u_height', 'desc_units', 'outer_width', 'outer_depth', 'outer_unit', 'comments', 'tags',
]
2016-03-01 11:23:03 -05:00
help_texts = {
'site': "The site at which the rack exists",
'name': "Organizational rack name",
'facility_id': "The unique rack ID assigned by the facility",
'u_height': "Height in rack units",
}
widgets = {
'site': APISelect(
api_url="/api/dcim/sites/",
filter_for={
'group': 'site_id',
2018-11-27 11:57:29 -05:00
}
),
'status': StaticSelect2(),
'role': APISelect(
api_url="/api/dcim/rack-roles/"
),
'type': StaticSelect2(),
'width': StaticSelect2(),
'outer_unit': StaticSelect2(),
2016-03-01 11:23:03 -05:00
}
class RackCSVForm(forms.ModelForm):
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
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
group_name = forms.CharField(
help_text='Name of rack group',
required=False
)
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.',
}
)
status = CSVChoiceField(
choices=RACK_STATUS_CHOICES,
required=False,
help_text='Operational status'
)
role = forms.ModelChoiceField(
queryset=RackRole.objects.all(),
required=False,
to_field_name='name',
help_text='Name of assigned role',
error_messages={
'invalid_choice': 'Role not found.',
}
)
type = CSVChoiceField(
choices=RACK_TYPE_CHOICES,
required=False,
help_text='Rack type'
)
width = forms.ChoiceField(
2017-06-07 15:56:59 -04:00
choices=(
(RACK_WIDTH_19IN, '19'),
(RACK_WIDTH_23IN, '23'),
),
help_text='Rail-to-rail width (in inches)'
)
outer_unit = CSVChoiceField(
choices=RACK_DIMENSION_UNIT_CHOICES,
required=False,
help_text='Unit for outer dimensions'
)
2016-03-01 11:23:03 -05:00
class Meta:
model = Rack
fields = Rack.csv_headers
help_texts = {
'name': 'Rack name',
'u_height': 'Height in rack units',
}
2016-03-01 11:23:03 -05:00
2017-06-07 14:19:08 -04:00
def clean(self):
super().clean()
2016-03-01 11:23:03 -05:00
site = self.cleaned_data.get('site')
2017-06-07 14:19:08 -04:00
group_name = self.cleaned_data.get('group_name')
name = self.cleaned_data.get('name')
facility_id = self.cleaned_data.get('facility_id')
2016-03-01 11:23:03 -05:00
2017-06-07 14:19:08 -04:00
# Validate rack group
if group_name:
try:
self.instance.group = RackGroup.objects.get(site=site, name=group_name)
except RackGroup.DoesNotExist:
raise forms.ValidationError("Rack group {} not found for site {}".format(group_name, site))
2016-03-01 11:23:03 -05:00
# Validate uniqueness of rack name within group
if Rack.objects.filter(group=self.instance.group, name=name).exists():
raise forms.ValidationError(
"A rack named {} already exists within group {}".format(name, group_name)
)
# Validate uniqueness of facility ID within group
if facility_id and Rack.objects.filter(group=self.instance.group, facility_id=facility_id).exists():
raise forms.ValidationError(
"A rack with the facility ID {} already exists within group {}".format(facility_id, group_name)
)
2016-03-01 11:23:03 -05:00
2018-07-10 10:00:21 -04:00
class RackBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
pk = forms.ModelMultipleChoiceField(
queryset=Rack.objects.all(),
widget=forms.MultipleHiddenInput
)
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
2019-01-08 15:35:34 -08:00
required=False,
widget=APISelect(
api_url="/api/dcim/sites",
filter_for={
'group': 'site_id',
}
)
)
group = forms.ModelChoiceField(
queryset=RackGroup.objects.all(),
2019-01-08 15:35:34 -08:00
required=False,
widget=APISelect(
api_url="/api/dcim/rack-groups",
)
)
tenant = forms.ModelChoiceField(
queryset=Tenant.objects.all(),
2019-01-08 15:35:34 -08:00
required=False,
widget=APISelect(
api_url="/api/tenancy/tenants",
)
)
status = forms.ChoiceField(
choices=add_blank_choice(RACK_STATUS_CHOICES),
required=False,
2019-01-08 15:35:34 -08:00
initial='',
widget=StaticSelect2()
)
role = forms.ModelChoiceField(
queryset=RackRole.objects.all(),
2019-01-08 15:35:34 -08:00
required=False,
widget=APISelect(
api_url="/api/dcim/rack-roles",
)
)
serial = forms.CharField(
max_length=50,
required=False,
label='Serial Number'
)
2018-11-02 09:17:51 -04:00
asset_tag = forms.CharField(
max_length=50,
required=False
)
type = forms.ChoiceField(
choices=add_blank_choice(RACK_TYPE_CHOICES),
2019-01-08 15:35:34 -08:00
required=False,
widget=StaticSelect2()
)
width = forms.ChoiceField(
choices=add_blank_choice(RACK_WIDTH_CHOICES),
2019-01-08 15:35:34 -08:00
required=False,
widget=StaticSelect2()
)
u_height = forms.IntegerField(
required=False,
label='Height (U)'
)
desc_units = forms.NullBooleanField(
required=False,
widget=BulkEditNullBooleanSelect,
label='Descending units'
)
outer_width = forms.IntegerField(
required=False,
min_value=1
)
outer_depth = forms.IntegerField(
required=False,
min_value=1
)
outer_unit = forms.ChoiceField(
choices=add_blank_choice(RACK_DIMENSION_UNIT_CHOICES),
2019-01-08 15:35:34 -08:00
required=False,
widget=StaticSelect2()
)
comments = CommentField(
widget=SmallTextarea
)
2016-03-01 11:23:03 -05:00
class Meta:
nullable_fields = [
'group', 'tenant', 'role', 'serial', 'asset_tag', 'outer_width', 'outer_depth', 'outer_unit', 'comments',
]
2016-03-01 11:23:03 -05:00
class RackFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldFilterForm):
model = Rack
2019-05-09 14:32:49 -04:00
field_order = ['q', 'site', 'group_id', 'status', 'role', 'tenant_group', 'tenant']
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
site = FilterChoiceField(
2019-01-09 13:44:39 -05:00
queryset=Site.objects.all(),
2019-01-08 15:35:34 -08:00
to_field_name='slug',
widget=APISelectMultiple(
api_url="/api/dcim/sites/",
value_field="slug",
filter_for={
'group_id': 'site'
}
2019-01-08 15:35:34 -08:00
)
)
group_id = ChainedModelChoiceField(
label='Rack group',
queryset=RackGroup.objects.prefetch_related('site'),
chains=(
('site', 'site'),
),
2019-06-24 15:48:49 -04:00
required=False,
2019-01-08 15:35:34 -08:00
widget=APISelectMultiple(
api_url="/api/dcim/rack-groups/",
null_option=True,
)
)
2019-01-09 23:33:08 -05:00
status = forms.MultipleChoiceField(
choices=RACK_STATUS_CHOICES,
2019-01-08 15:35:34 -08:00
required=False,
widget=StaticSelect2Multiple()
)
role = FilterChoiceField(
2019-01-09 13:44:39 -05:00
queryset=RackRole.objects.all(),
to_field_name='slug',
2019-01-08 15:35:34 -08:00
null_label='-- None --',
widget=APISelectMultiple(
api_url="/api/dcim/rack-roles/",
value_field="slug",
null_option=True,
)
)
2016-03-01 11:23:03 -05:00
#
# Rack reservations
#
class RackReservationForm(BootstrapMixin, TenancyForm, forms.ModelForm):
2018-11-27 11:57:29 -05:00
units = SimpleArrayField(
base_field=forms.IntegerField(),
widget=ArrayFieldSelectMultiple(
attrs={
'size': 10,
}
)
)
user = forms.ModelChoiceField(
queryset=User.objects.order_by(
'username'
2019-01-08 15:35:34 -08:00
),
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
class Meta:
model = RackReservation
2018-11-27 11:57:29 -05:00
fields = [
'units', 'user', 'tenant_group', 'tenant', 'description',
]
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# Populate rack unit choices
self.fields['units'].widget.choices = self._get_unit_choices()
def _get_unit_choices(self):
rack = self.instance.rack
reserved_units = []
for resv in rack.reservations.exclude(pk=self.instance.pk):
for u in resv.units:
reserved_units.append(u)
unit_choices = [(u, {'label': str(u), 'disabled': u in reserved_units}) for u in rack.units]
return unit_choices
class RackReservationBulkEditForm(BootstrapMixin, BulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=RackReservation.objects.all(),
widget=forms.MultipleHiddenInput()
)
user = forms.ModelChoiceField(
queryset=User.objects.order_by(
'username'
),
2019-01-08 15:35:34 -08:00
required=False,
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
tenant = forms.ModelChoiceField(
queryset=Tenant.objects.all(),
2019-01-08 15:35:34 -08:00
required=False,
widget=APISelect(
api_url="/api/tenancy/tenant",
)
2018-11-27 11:57:29 -05:00
)
description = forms.CharField(
max_length=100,
required=False
)
class Meta:
nullable_fields = []
2019-05-09 14:32:49 -04:00
class RackReservationFilterForm(BootstrapMixin, TenancyFilterForm):
field_order = ['q', 'site', 'group_id', 'tenant_group', 'tenant']
q = forms.CharField(
required=False,
label='Search'
)
site = FilterChoiceField(
queryset=Site.objects.all(),
to_field_name='slug',
widget=APISelectMultiple(
api_url="/api/dcim/sites/",
value_field="slug",
)
)
group_id = FilterChoiceField(
queryset=RackGroup.objects.prefetch_related('site'),
2019-05-09 14:32:49 -04:00
label='Rack group',
null_label='-- None --',
widget=APISelectMultiple(
api_url="/api/dcim/rack-groups/",
null_option=True,
)
)
2016-05-13 15:22:31 -04:00
#
# Manufacturers
#
class ManufacturerForm(BootstrapMixin, forms.ModelForm):
2016-05-20 15:32:17 -04:00
slug = SlugField()
2016-05-13 15:22:31 -04:00
class Meta:
model = Manufacturer
2018-11-27 11:57:29 -05:00
fields = [
'name', 'slug',
]
2016-05-13 15:22:31 -04:00
class ManufacturerCSVForm(forms.ModelForm):
2018-11-27 11:57:29 -05:00
class Meta:
model = Manufacturer
fields = Manufacturer.csv_headers
help_texts = {
'name': 'Manufacturer name',
'slug': 'URL-friendly slug',
}
#
# Device types
#
class DeviceTypeForm(BootstrapMixin, CustomFieldForm):
2018-11-27 11:57:29 -05:00
slug = SlugField(
slug_source='model'
)
comments = CommentField()
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
class Meta:
model = DeviceType
fields = [
'manufacturer', 'model', 'slug', 'part_number', 'u_height', 'is_full_depth', 'subdevice_role', 'comments',
'tags',
]
widgets = {
'manufacturer': APISelect(
api_url="/api/dcim/manufacturers/"
),
'subdevice_role': StaticSelect2()
}
class DeviceTypeImportForm(BootstrapMixin, forms.ModelForm):
manufacturer = forms.ModelChoiceField(
queryset=Manufacturer.objects.all(),
to_field_name='name'
)
class Meta:
model = DeviceType
fields = [
'manufacturer', 'model', 'slug', 'part_number', 'u_height', 'is_full_depth', 'subdevice_role',
]
2018-07-10 10:00:21 -04:00
class DeviceTypeBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=DeviceType.objects.all(),
widget=forms.MultipleHiddenInput()
)
manufacturer = forms.ModelChoiceField(
queryset=Manufacturer.objects.all(),
2019-01-08 15:35:34 -08:00
required=False,
widget=APISelect(
api_url="/api/dcim/manufactureres"
)
2018-11-27 11:57:29 -05:00
)
u_height = forms.IntegerField(
min_value=1,
required=False
)
2018-11-27 11:57:29 -05:00
is_full_depth = forms.NullBooleanField(
required=False,
widget=BulkEditNullBooleanSelect(),
label='Is full depth'
)
class Meta:
nullable_fields = []
class DeviceTypeFilterForm(BootstrapMixin, CustomFieldFilterForm):
model = DeviceType
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
manufacturer = FilterChoiceField(
2019-01-09 13:44:39 -05:00
queryset=Manufacturer.objects.all(),
2019-01-08 15:35:34 -08:00
to_field_name='slug',
widget=APISelectMultiple(
api_url="/api/dcim/manufacturers/",
value_field="slug",
)
)
subdevice_role = forms.NullBooleanField(
required=False,
label='Subdevice role',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=add_blank_choice(SUBDEVICE_ROLE_CHOICES)
)
)
console_ports = forms.NullBooleanField(
required=False,
label='Has console ports',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
console_server_ports = forms.NullBooleanField(
required=False,
label='Has console server ports',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
power_ports = forms.NullBooleanField(
required=False,
label='Has power ports',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
power_outlets = forms.NullBooleanField(
required=False,
label='Has power outlets',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
interfaces = forms.NullBooleanField(
required=False,
label='Has interfaces',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
pass_through_ports = forms.NullBooleanField(
required=False,
label='Has pass-through ports',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
#
# Device component templates
#
class ConsolePortTemplateForm(BootstrapMixin, forms.ModelForm):
class Meta:
model = ConsolePortTemplate
2018-11-27 11:57:29 -05:00
fields = [
'device_type', 'name', 'type',
2018-11-27 11:57:29 -05:00
]
widgets = {
'device_type': forms.HiddenInput(),
}
class ConsolePortTemplateCreateForm(ComponentForm):
2018-11-27 11:57:29 -05:00
name_pattern = ExpandableNameField(
label='Name'
)
type = forms.ChoiceField(
choices=CONSOLE_TYPE_CHOICES,
widget=StaticSelect2()
)
class ConsoleServerPortTemplateForm(BootstrapMixin, forms.ModelForm):
class Meta:
model = ConsoleServerPortTemplate
2018-11-27 11:57:29 -05:00
fields = [
'device_type', 'name', 'type',
2018-11-27 11:57:29 -05:00
]
widgets = {
'device_type': forms.HiddenInput(),
}
class ConsoleServerPortTemplateCreateForm(ComponentForm):
2018-11-27 11:57:29 -05:00
name_pattern = ExpandableNameField(
label='Name'
)
type = forms.ChoiceField(
choices=CONSOLE_TYPE_CHOICES,
widget=StaticSelect2()
)
class PowerPortTemplateForm(BootstrapMixin, forms.ModelForm):
class Meta:
model = PowerPortTemplate
2018-11-27 11:57:29 -05:00
fields = [
'device_type', 'name', 'maximum_draw', 'allocated_draw',
2018-11-27 11:57:29 -05:00
]
widgets = {
'device_type': forms.HiddenInput(),
}
class PowerPortTemplateCreateForm(ComponentForm):
2018-11-27 11:57:29 -05:00
name_pattern = ExpandableNameField(
label='Name'
)
maximum_draw = forms.IntegerField(
min_value=1,
required=False,
help_text="Maximum current draw (watts)"
)
allocated_draw = forms.IntegerField(
min_value=1,
required=False,
help_text="Allocated current draw (watts)"
)
class PowerOutletTemplateForm(BootstrapMixin, forms.ModelForm):
class Meta:
model = PowerOutletTemplate
2018-11-27 11:57:29 -05:00
fields = [
'device_type', 'name', 'power_port', 'feed_leg',
2018-11-27 11:57:29 -05:00
]
widgets = {
'device_type': forms.HiddenInput(),
}
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# Limit power_port choices to current DeviceType
if hasattr(self.instance, 'device_type'):
self.fields['power_port'].queryset = PowerPortTemplate.objects.filter(
device_type=self.instance.device_type
)
class PowerOutletTemplateCreateForm(ComponentForm):
name_pattern = ExpandableNameField(
label='Name'
)
power_port = forms.ModelChoiceField(
queryset=PowerPortTemplate.objects.all(),
required=False
)
feed_leg = forms.ChoiceField(
choices=add_blank_choice(POWERFEED_LEG_CHOICES),
required=False,
widget=StaticSelect2()
)
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# Limit power_port choices to current DeviceType
self.fields['power_port'].queryset = PowerPortTemplate.objects.filter(
device_type=self.parent
)
class InterfaceTemplateForm(BootstrapMixin, forms.ModelForm):
class Meta:
model = InterfaceTemplate
2018-11-27 11:57:29 -05:00
fields = [
'device_type', 'name', 'type', 'mgmt_only',
2018-11-27 11:57:29 -05:00
]
widgets = {
'device_type': forms.HiddenInput(),
'type': StaticSelect2(),
}
class InterfaceTemplateCreateForm(ComponentForm):
2018-11-27 11:57:29 -05:00
name_pattern = ExpandableNameField(
label='Name'
)
type = forms.ChoiceField(
choices=IFACE_TYPE_CHOICES,
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
mgmt_only = forms.BooleanField(
required=False,
label='Management only'
)
class InterfaceTemplateBulkEditForm(BootstrapMixin, BulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=InterfaceTemplate.objects.all(),
widget=forms.MultipleHiddenInput()
)
type = forms.ChoiceField(
choices=add_blank_choice(IFACE_TYPE_CHOICES),
2019-01-08 15:35:34 -08:00
required=False,
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
mgmt_only = forms.NullBooleanField(
required=False,
widget=BulkEditNullBooleanSelect,
label='Management only'
)
class Meta:
nullable_fields = []
class FrontPortTemplateForm(BootstrapMixin, forms.ModelForm):
2018-10-03 14:04:16 -04:00
class Meta:
model = FrontPortTemplate
2018-11-27 11:57:29 -05:00
fields = [
'device_type', 'name', 'type', 'rear_port', 'rear_port_position',
]
2018-10-03 14:04:16 -04:00
widgets = {
'device_type': forms.HiddenInput(),
'rear_port': StaticSelect2(),
2018-10-03 14:04:16 -04:00
}
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# Limit rear_port choices to current DeviceType
if hasattr(self.instance, 'device_type'):
self.fields['rear_port'].queryset = RearPortTemplate.objects.filter(
device_type=self.instance.device_type
)
2018-10-03 14:04:16 -04:00
class FrontPortTemplateCreateForm(ComponentForm):
2018-10-03 14:04:16 -04:00
name_pattern = ExpandableNameField(
label='Name'
)
type = forms.ChoiceField(
2019-01-08 15:35:34 -08:00
choices=PORT_TYPE_CHOICES,
widget=StaticSelect2()
2018-10-03 14:04:16 -04:00
)
rear_port_set = forms.MultipleChoiceField(
choices=[],
label='Rear ports',
help_text='Select one rear port assignment for each front port being created.',
2018-10-03 14:04:16 -04:00
)
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
2018-10-03 14:04:16 -04:00
# Determine which rear port positions are occupied. These will be excluded from the list of available mappings.
occupied_port_positions = [
(front_port.rear_port_id, front_port.rear_port_position)
for front_port in self.parent.frontport_templates.all()
2018-10-03 14:04:16 -04:00
]
# Populate rear port choices
choices = []
rear_ports = RearPortTemplate.objects.filter(device_type=self.parent)
2018-10-03 14:04:16 -04:00
for rear_port in rear_ports:
for i in range(1, rear_port.positions + 1):
if (rear_port.pk, i) not in occupied_port_positions:
choices.append(
('{}:{}'.format(rear_port.pk, i), '{}:{}'.format(rear_port.name, i))
)
self.fields['rear_port_set'].choices = choices
def clean(self):
# Validate that the number of ports being created equals the number of selected (rear port, position) tuples
front_port_count = len(self.cleaned_data['name_pattern'])
rear_port_count = len(self.cleaned_data['rear_port_set'])
if front_port_count != rear_port_count:
raise forms.ValidationError({
'rear_port_set': 'The provided name pattern will create {} ports, however {} rear port assignments '
'were selected. These counts must match.'.format(front_port_count, rear_port_count)
})
def get_iterative_data(self, iteration):
# Assign rear port and position from selected set
rear_port, position = self.cleaned_data['rear_port_set'][iteration].split(':')
return {
'rear_port': int(rear_port),
'rear_port_position': int(position),
}
class RearPortTemplateForm(BootstrapMixin, forms.ModelForm):
2018-10-03 14:04:16 -04:00
class Meta:
model = RearPortTemplate
2018-11-27 11:57:29 -05:00
fields = [
'device_type', 'name', 'type', 'positions',
]
2018-10-03 14:04:16 -04:00
widgets = {
'device_type': forms.HiddenInput(),
'type': StaticSelect2(),
2018-10-03 14:04:16 -04:00
}
class RearPortTemplateCreateForm(ComponentForm):
2018-10-03 14:04:16 -04:00
name_pattern = ExpandableNameField(
label='Name'
)
type = forms.ChoiceField(
choices=PORT_TYPE_CHOICES,
widget=StaticSelect2(),
2018-10-03 14:04:16 -04:00
)
positions = forms.IntegerField(
min_value=1,
max_value=64,
initial=1,
help_text='The number of front ports which may be mapped to each rear port'
)
class DeviceBayTemplateForm(BootstrapMixin, forms.ModelForm):
class Meta:
model = DeviceBayTemplate
2018-11-27 11:57:29 -05:00
fields = [
'device_type', 'name',
]
widgets = {
'device_type': forms.HiddenInput(),
}
class DeviceBayTemplateCreateForm(ComponentForm):
2018-11-27 11:57:29 -05:00
name_pattern = ExpandableNameField(
label='Name'
)
#
# Component template import forms
#
class ComponentTemplateImportForm(BootstrapMixin, forms.ModelForm):
def __init__(self, device_type, data=None, *args, **kwargs):
# Must pass the parent DeviceType on form initialization
data.update({
'device_type': device_type.pk,
})
super().__init__(data, *args, **kwargs)
def clean_device_type(self):
data = self.cleaned_data['device_type']
# Limit fields referencing other components to the parent DeviceType
for field_name, field in self.fields.items():
if isinstance(field, forms.ModelChoiceField) and field_name != 'device_type':
field.queryset = field.queryset.filter(device_type=data)
return data
class ConsolePortTemplateImportForm(ComponentTemplateImportForm):
type = forms.ChoiceField(
choices=ConsolePortTypes.TYPE_CHOICES
)
class Meta:
model = ConsolePortTemplate
fields = [
'device_type', 'name', 'type',
]
def clean_type(self):
# Convert slug value to field integer value
slug = self.cleaned_data['type']
return ConsolePortTypes.slug_to_integer(slug)
class ConsoleServerPortTemplateImportForm(ComponentTemplateImportForm):
type = forms.ChoiceField(
choices=ConsolePortTypes.TYPE_CHOICES
)
class Meta:
model = ConsoleServerPortTemplate
fields = [
'device_type', 'name', 'type',
]
def clean_type(self):
# Convert slug value to field integer value
slug = self.cleaned_data['type']
return ConsolePortTypes.slug_to_integer(slug)
class PowerPortTemplateImportForm(ComponentTemplateImportForm):
class Meta:
model = PowerPortTemplate
fields = [
'device_type', 'name', 'maximum_draw', 'allocated_draw',
]
class PowerOutletTemplateImportForm(ComponentTemplateImportForm):
power_port = forms.ModelChoiceField(
queryset=PowerPortTemplate.objects.all(),
to_field_name='name',
required=False
)
class Meta:
model = PowerOutletTemplate
fields = [
'device_type', 'name', 'power_port', 'feed_leg',
]
class InterfaceTemplateImportForm(ComponentTemplateImportForm):
type = forms.ChoiceField(
choices=InterfaceTypes.TYPE_CHOICES
)
class Meta:
model = InterfaceTemplate
fields = [
'device_type', 'name', 'type', 'mgmt_only',
]
def clean_type(self):
# Convert slug value to field integer value
slug = self.cleaned_data['type']
return InterfaceTypes.slug_to_integer(slug)
class FrontPortTemplateImportForm(ComponentTemplateImportForm):
type = forms.ChoiceField(
choices=PortTypes.TYPE_CHOICES
)
2019-10-01 16:36:31 -04:00
rear_port = forms.ModelChoiceField(
queryset=RearPortTemplate.objects.all(),
to_field_name='name',
required=False
)
class Meta:
model = FrontPortTemplate
fields = [
'device_type', 'name', 'type', 'rear_port', 'rear_port_position',
]
def clean_type(self):
# Convert slug value to field integer value
slug = self.cleaned_data['type']
return PortTypes.slug_to_integer(slug)
class RearPortTemplateImportForm(ComponentTemplateImportForm):
type = forms.ChoiceField(
choices=PortTypes.TYPE_CHOICES
)
class Meta:
model = RearPortTemplate
fields = [
'device_type', 'name', 'type', 'positions',
]
def clean_type(self):
# Convert slug value to field integer value
slug = self.cleaned_data['type']
return PortTypes.slug_to_integer(slug)
class DeviceBayTemplateImportForm(ComponentTemplateImportForm):
class Meta:
model = DeviceBayTemplate
fields = [
'device_type', 'name',
]
2016-05-12 14:38:34 -04:00
#
# Device roles
#
class DeviceRoleForm(BootstrapMixin, forms.ModelForm):
2016-05-20 15:32:17 -04:00
slug = SlugField()
2016-05-12 14:38:34 -04:00
class Meta:
model = DeviceRole
2018-11-27 11:57:29 -05:00
fields = [
'name', 'slug', 'color', 'vm_role',
]
2016-05-12 14:38:34 -04:00
class DeviceRoleCSVForm(forms.ModelForm):
slug = SlugField()
class Meta:
model = DeviceRole
fields = DeviceRole.csv_headers
help_texts = {
'name': 'Name of device role',
'color': 'RGB color in hexadecimal (e.g. 00ff00)'
}
2016-05-16 11:54:17 -04:00
#
# Platforms
#
class PlatformForm(BootstrapMixin, forms.ModelForm):
slug = SlugField(
max_length=64
)
2016-05-16 11:54:17 -04:00
class Meta:
model = Platform
2018-11-27 11:57:29 -05:00
fields = [
'name', 'slug', 'manufacturer', 'napalm_driver', 'napalm_args',
]
widgets = {
'manufacturer': APISelect(
api_url="/api/dcim/manufacturers/"
),
'napalm_args': SmallTextarea(),
}
2016-05-16 11:54:17 -04:00
class PlatformCSVForm(forms.ModelForm):
slug = SlugField()
manufacturer = forms.ModelChoiceField(
queryset=Manufacturer.objects.all(),
required=False,
to_field_name='name',
help_text='Manufacturer name',
error_messages={
'invalid_choice': 'Manufacturer not found.',
}
)
class Meta:
model = Platform
fields = Platform.csv_headers
help_texts = {
'name': 'Platform name',
}
2016-03-01 11:23:03 -05:00
#
# Devices
#
class DeviceForm(BootstrapMixin, TenancyForm, CustomFieldForm):
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
widget=APISelect(
api_url="/api/dcim/sites/",
filter_for={
'rack': 'site_id'
2018-11-27 11:57:29 -05:00
}
)
)
rack = ChainedModelChoiceField(
queryset=Rack.objects.all(),
chains=(
('site', 'site'),
),
required=False,
widget=APISelect(
api_url='/api/dcim/racks/',
display_field='display_name'
)
)
position = forms.TypedChoiceField(
required=False,
empty_value=None,
help_text="The lowest-numbered unit occupied by the device",
widget=APISelect(
api_url='/api/dcim/racks/{{rack}}/units/',
disabled_indicator='device'
)
)
manufacturer = forms.ModelChoiceField(
queryset=Manufacturer.objects.all(),
widget=APISelect(
api_url="/api/dcim/manufacturers/",
filter_for={
'device_type': 'manufacturer_id'
2018-11-27 11:57:29 -05:00
}
)
)
device_type = ChainedModelChoiceField(
queryset=DeviceType.objects.all(),
chains=(
('manufacturer', 'manufacturer'),
),
label='Device type',
widget=APISelect(
api_url='/api/dcim/device-types/',
display_field='model'
)
)
cluster_group = forms.ModelChoiceField(
queryset=ClusterGroup.objects.all(),
required=False,
widget=APISelect(
api_url="/api/virtualization/cluster-groups/",
filter_for={
'cluster': 'group_id'
},
attrs={
'nullable': 'true'
}
)
)
cluster = ChainedModelChoiceField(
queryset=Cluster.objects.all(),
chains=(
('group', 'cluster_group'),
),
required=False,
widget=APISelect(
api_url='/api/virtualization/clusters/',
)
)
2016-03-01 11:23:03 -05:00
comments = CommentField()
tags = TagField(required=False)
local_context_data = JSONField(
required=False,
label=''
)
2016-03-01 11:23:03 -05:00
class Meta:
model = Device
fields = [
'name', 'device_role', 'device_type', 'serial', 'asset_tag', 'site', 'rack', 'position', 'face',
'status', 'platform', 'primary_ip4', 'primary_ip6', 'cluster_group', 'cluster', 'tenant_group', 'tenant',
'comments', 'tags', 'local_context_data'
]
2016-03-01 11:23:03 -05:00
help_texts = {
'device_role': "The function this device serves",
'serial': "Chassis serial number",
2018-11-27 11:57:29 -05:00
'local_context_data': "Local config context data overwrites all source contexts in the final rendered "
"config context",
2016-03-01 11:23:03 -05:00
}
widgets = {
'face': StaticSelect2(
filter_for={
'position': 'face'
2018-11-27 11:57:29 -05:00
}
),
'device_role': APISelect(
api_url='/api/dcim/device-roles/'
),
'status': StaticSelect2(),
'platform': APISelect(
api_url="/api/dcim/platforms/"
),
'primary_ip4': StaticSelect2(),
'primary_ip6': 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')
if 'initial' not in kwargs:
kwargs['initial'] = {}
# Using hasattr() instead of "is not None" to avoid RelatedObjectDoesNotExist on required field
if instance and hasattr(instance, 'device_type'):
kwargs['initial']['manufacturer'] = instance.device_type.manufacturer
if instance and instance.cluster is not None:
kwargs['initial']['cluster_group'] = instance.cluster.group
2016-03-01 11:23:03 -05:00
super().__init__(*args, **kwargs)
2016-03-01 11:23:03 -05:00
if self.instance.pk:
2016-03-01 11:23:03 -05:00
# 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 Device or a peer VirtualChassis member
interface_ids = self.instance.vc_interfaces.values('pk')
# Collect interface IPs
interface_ips = IPAddress.objects.prefetch_related('interface').filter(
family=family, interface_id__in=interface_ids
)
if interface_ips:
ip_list = [(ip.id, '{} ({})'.format(ip.address, ip.interface)) for ip in interface_ips]
ip_choices.append(('Interface IPs', ip_list))
# Collect NAT IPs
nat_ips = IPAddress.objects.prefetch_related('nat_inside').filter(
family=family, nat_inside__interface__in=interface_ids
)
if nat_ips:
ip_list = [(ip.id, '{} ({})'.format(ip.address, ip.nat_inside.address)) for ip in nat_ips]
ip_choices.append(('NAT IPs', ip_list))
self.fields['primary_ip{}'.format(family)].choices = ip_choices
2016-03-01 11:23:03 -05:00
# If editing an existing device, exclude it from the list of occupied rack units. This ensures that a device
# can be flipped from one face to another.
2018-12-30 02:35:18 -05:00
self.fields['position'].widget.add_additional_query_param('exclude', self.instance.pk)
# Limit platform by manufacturer
self.fields['platform'].queryset = Platform.objects.filter(
Q(manufacturer__isnull=True) | Q(manufacturer=self.instance.device_type.manufacturer)
)
2016-03-01 11:23:03 -05:00
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
2016-03-01 11:23:03 -05:00
# Rack position
pk = self.instance.pk if self.instance.pk else None
2016-03-01 11:23:03 -05:00
try:
2016-05-18 13:19:26 -04:00
if self.is_bound and self.data.get('rack') and str(self.data.get('face')):
position_choices = Rack.objects.get(pk=self.data['rack']) \
.get_rack_units(face=self.data.get('face'), exclude=pk)
2016-05-18 13:19:26 -04:00
elif self.initial.get('rack') and str(self.initial.get('face')):
position_choices = Rack.objects.get(pk=self.initial['rack']) \
.get_rack_units(face=self.initial.get('face'), exclude=pk)
2016-03-01 11:23:03 -05:00
else:
position_choices = []
except Rack.DoesNotExist:
position_choices = []
self.fields['position'].choices = [('', '---------')] + [
(p['id'], {
'label': p['name'],
'disabled': bool(p['device'] and p['id'] != self.initial.get('position')),
}) for p in position_choices
]
# Disable rack assignment if this is a child device installed in a parent device
if pk and self.instance.device_type.is_child_device and hasattr(self.instance, 'parent_bay'):
self.fields['site'].disabled = True
self.fields['rack'].disabled = True
self.initial['site'] = self.instance.parent_bay.device.site_id
self.initial['rack'] = self.instance.parent_bay.device.rack_id
2016-03-01 11:23:03 -05:00
class BaseDeviceCSVForm(forms.ModelForm):
device_role = forms.ModelChoiceField(
queryset=DeviceRole.objects.all(),
to_field_name='name',
help_text='Name of assigned role',
error_messages={
'invalid_choice': 'Invalid device role.',
}
)
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.',
}
)
manufacturer = forms.ModelChoiceField(
queryset=Manufacturer.objects.all(),
to_field_name='name',
help_text='Device type manufacturer',
error_messages={
'invalid_choice': 'Invalid manufacturer.',
}
)
model_name = forms.CharField(
help_text='Device type model name'
)
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.',
}
)
status = CSVChoiceField(
choices=DEVICE_STATUS_CHOICES,
help_text='Operational status'
)
2016-03-01 11:23:03 -05:00
class Meta:
fields = []
2016-03-01 11:23:03 -05:00
model = Device
help_texts = {
'name': 'Device name',
}
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
manufacturer = self.cleaned_data.get('manufacturer')
model_name = self.cleaned_data.get('model_name')
# Validate device type
if manufacturer and model_name:
try:
self.instance.device_type = DeviceType.objects.get(manufacturer=manufacturer, model=model_name)
except DeviceType.DoesNotExist:
2017-06-07 14:19:08 -04:00
raise forms.ValidationError("Device type {} {} not found".format(manufacturer, model_name))
2016-03-01 11:23:03 -05:00
class DeviceCSVForm(BaseDeviceCSVForm):
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
to_field_name='name',
help_text='Name of parent site',
error_messages={
'invalid_choice': 'Invalid site name.',
}
)
2017-06-07 14:19:08 -04:00
rack_group = forms.CharField(
required=False,
help_text='Parent rack\'s group (if any)'
2017-06-07 14:19:08 -04:00
)
rack_name = forms.CharField(
required=False,
help_text='Name of parent rack'
)
face = CSVChoiceField(
choices=RACK_FACE_CHOICES,
required=False,
help_text='Mounted rack face'
)
cluster = forms.ModelChoiceField(
queryset=Cluster.objects.all(),
to_field_name='name',
required=False,
help_text='Virtualization cluster',
error_messages={
'invalid_choice': 'Invalid cluster name.',
}
)
class Meta(BaseDeviceCSVForm.Meta):
fields = [
'name', 'device_role', 'tenant', 'manufacturer', 'model_name', 'platform', 'serial', 'asset_tag', 'status',
'site', 'rack_group', 'rack_name', 'position', 'face', 'cluster', 'comments',
]
def clean(self):
super().clean()
site = self.cleaned_data.get('site')
2017-06-07 14:19:08 -04:00
rack_group = self.cleaned_data.get('rack_group')
rack_name = self.cleaned_data.get('rack_name')
2016-03-01 11:23:03 -05:00
# Validate rack
2017-06-07 14:19:08 -04:00
if site and rack_group and rack_name:
2016-03-01 11:23:03 -05:00
try:
2017-06-07 14:19:08 -04:00
self.instance.rack = Rack.objects.get(site=site, group__name=rack_group, name=rack_name)
2016-03-01 11:23:03 -05:00
except Rack.DoesNotExist:
2017-06-07 14:19:08 -04:00
raise forms.ValidationError("Rack {} not found in site {} group {}".format(rack_name, site, rack_group))
elif site and rack_name:
try:
self.instance.rack = Rack.objects.get(site=site, group__isnull=True, name=rack_name)
except Rack.DoesNotExist:
raise forms.ValidationError("Rack {} not found in site {} (no group)".format(rack_name, site))
class ChildDeviceCSVForm(BaseDeviceCSVForm):
parent = FlexibleModelChoiceField(
queryset=Device.objects.all(),
to_field_name='name',
help_text='Name or ID of parent device',
error_messages={
'invalid_choice': 'Parent device not found.',
}
)
device_bay_name = forms.CharField(
help_text='Name of device bay',
)
cluster = forms.ModelChoiceField(
queryset=Cluster.objects.all(),
to_field_name='name',
required=False,
help_text='Virtualization cluster',
error_messages={
'invalid_choice': 'Invalid cluster name.',
}
)
class Meta(BaseDeviceCSVForm.Meta):
fields = [
'name', 'device_role', 'tenant', 'manufacturer', 'model_name', 'platform', 'serial', 'asset_tag', 'status',
'parent', 'device_bay_name', 'cluster', 'comments',
]
def clean(self):
super().clean()
parent = self.cleaned_data.get('parent')
device_bay_name = self.cleaned_data.get('device_bay_name')
# Validate device bay
if parent and device_bay_name:
try:
2017-06-07 14:19:08 -04:00
self.instance.parent_bay = DeviceBay.objects.get(device=parent, name=device_bay_name)
# Inherit site and rack from parent device
self.instance.site = parent.site
self.instance.rack = parent.rack
except DeviceBay.DoesNotExist:
raise forms.ValidationError("Parent device/bay ({} {}) not found".format(parent, device_bay_name))
2018-07-10 10:00:21 -04:00
class DeviceBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=Device.objects.all(),
widget=forms.MultipleHiddenInput()
)
2018-11-27 11:57:29 -05:00
device_type = forms.ModelChoiceField(
queryset=DeviceType.objects.all(),
required=False,
2019-01-08 15:35:34 -08:00
label='Type',
widget=APISelect(
api_url="/api/dcim/device-types/",
display_field='display_name'
2019-01-08 15:35:34 -08:00
)
2018-11-27 11:57:29 -05:00
)
device_role = forms.ModelChoiceField(
queryset=DeviceRole.objects.all(),
required=False,
2019-01-08 15:35:34 -08:00
label='Role',
widget=APISelect(
2019-01-10 21:19:13 -05:00
api_url="/api/dcim/device-roles/"
2019-01-08 15:35:34 -08:00
)
2018-11-27 11:57:29 -05:00
)
tenant = forms.ModelChoiceField(
queryset=Tenant.objects.all(),
2019-01-08 15:35:34 -08:00
required=False,
widget=APISelect(
2019-01-10 21:19:13 -05:00
api_url="/api/tenancy/tenants/"
2019-01-08 15:35:34 -08:00
)
2018-11-27 11:57:29 -05:00
)
platform = forms.ModelChoiceField(
queryset=Platform.objects.all(),
2019-01-08 15:35:34 -08:00
required=False,
widget=APISelect(
2019-01-10 21:19:13 -05:00
api_url="/api/dcim/platforms/"
2019-01-08 15:35:34 -08:00
)
2018-11-27 11:57:29 -05:00
)
status = forms.ChoiceField(
choices=add_blank_choice(DEVICE_STATUS_CHOICES),
required=False,
2019-01-08 15:35:34 -08:00
initial='',
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
serial = forms.CharField(
max_length=50,
required=False,
label='Serial Number'
)
2016-03-01 11:23:03 -05:00
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'tenant', 'platform', 'serial',
]
2016-03-01 11:23:03 -05:00
class DeviceFilterForm(BootstrapMixin, LocalConfigContextFilterForm, TenancyFilterForm, CustomFieldFilterForm):
model = Device
2019-05-09 14:32:49 -04:00
field_order = [
'q', 'region', 'site', 'rack_group_id', 'rack_id', 'status', 'role', 'tenant_group', 'tenant',
'manufacturer_id', 'device_type_id', 'mac_address', 'has_primary_ip',
]
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
2019-01-08 15:35:34 -08:00
region = FilterChoiceField(
queryset=Region.objects.all(),
to_field_name='slug',
required=False,
2019-01-08 15:35:34 -08:00
widget=APISelectMultiple(
api_url="/api/dcim/regions/",
value_field="slug",
filter_for={
'site': 'region'
}
)
)
site = FilterChoiceField(
2019-01-09 13:44:39 -05:00
queryset=Site.objects.all(),
to_field_name='slug',
2019-01-08 15:35:34 -08:00
widget=APISelectMultiple(
api_url="/api/dcim/sites/",
value_field="slug",
filter_for={
'rack_group_id': 'site',
'rack_id': 'site',
}
)
)
rack_group_id = FilterChoiceField(
queryset=RackGroup.objects.prefetch_related(
2018-11-27 11:57:29 -05:00
'site'
),
label='Rack group',
2019-01-08 15:35:34 -08:00
widget=APISelectMultiple(
api_url="/api/dcim/rack-groups/",
filter_for={
'rack_id': 'rack_group_id',
}
)
)
rack_id = FilterChoiceField(
2019-01-09 13:44:39 -05:00
queryset=Rack.objects.all(),
label='Rack',
null_label='-- None --',
2019-01-08 15:35:34 -08:00
widget=APISelectMultiple(
api_url="/api/dcim/racks/",
null_option=True,
)
)
role = FilterChoiceField(
2019-01-09 13:44:39 -05:00
queryset=DeviceRole.objects.all(),
to_field_name='slug',
2019-01-08 15:35:34 -08:00
widget=APISelectMultiple(
api_url="/api/dcim/device-roles/",
value_field="slug",
)
)
2018-11-27 11:57:29 -05:00
manufacturer_id = FilterChoiceField(
queryset=Manufacturer.objects.all(),
2019-01-08 15:35:34 -08:00
label='Manufacturer',
widget=APISelectMultiple(
api_url="/api/dcim/manufacturers/",
filter_for={
'device_type_id': 'manufacturer_id',
}
)
2018-11-27 11:57:29 -05:00
)
device_type_id = FilterChoiceField(
queryset=DeviceType.objects.prefetch_related(
2018-11-27 11:57:29 -05:00
'manufacturer'
),
label='Model',
2019-01-08 15:35:34 -08:00
widget=APISelectMultiple(
api_url="/api/dcim/device-types/",
display_field="model",
)
)
platform = FilterChoiceField(
2019-01-09 13:44:39 -05:00
queryset=Platform.objects.all(),
to_field_name='slug',
null_label='-- None --',
2019-01-08 15:35:34 -08:00
widget=APISelectMultiple(
api_url="/api/dcim/platforms/",
value_field="slug",
null_option=True,
)
)
2019-01-09 23:33:08 -05:00
status = forms.MultipleChoiceField(
choices=DEVICE_STATUS_CHOICES,
2019-01-08 15:35:34 -08:00
required=False,
widget=StaticSelect2Multiple()
)
2018-11-27 11:57:29 -05:00
mac_address = forms.CharField(
required=False,
label='MAC address'
)
has_primary_ip = forms.NullBooleanField(
required=False,
label='Has a primary IP',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
virtual_chassis_member = forms.NullBooleanField(
required=False,
label='Virtual chassis member',
widget=StaticSelect2(
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
console_ports = forms.NullBooleanField(
required=False,
label='Has console ports',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
console_server_ports = forms.NullBooleanField(
required=False,
label='Has console server ports',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
power_ports = forms.NullBooleanField(
required=False,
label='Has power ports',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
power_outlets = forms.NullBooleanField(
required=False,
label='Has power outlets',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
interfaces = forms.NullBooleanField(
required=False,
label='Has interfaces',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
pass_through_ports = forms.NullBooleanField(
required=False,
label='Has pass-through ports',
2019-01-08 15:35:34 -08:00
widget=StaticSelect2(
2018-11-27 11:57:29 -05:00
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
2016-03-01 11:23:03 -05:00
#
# Bulk device component creation
#
class DeviceBulkAddComponentForm(BootstrapMixin, forms.Form):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=Device.objects.all(),
widget=forms.MultipleHiddenInput()
)
name_pattern = ExpandableNameField(
label='Name'
)
class DeviceBulkAddInterfaceForm(DeviceBulkAddComponentForm):
type = forms.ChoiceField(
choices=IFACE_TYPE_CHOICES,
2019-01-08 15:35:34 -08:00
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
enabled = forms.BooleanField(
required=False,
initial=True
)
mtu = forms.IntegerField(
required=False,
min_value=1,
max_value=32767,
label='MTU'
)
mgmt_only = forms.BooleanField(
required=False,
label='Management only'
)
description = forms.CharField(
max_length=100,
required=False
)
2016-03-01 11:23:03 -05:00
#
# Console ports
#
class ConsolePortForm(BootstrapMixin, forms.ModelForm):
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
model = ConsolePort
2018-11-27 11:57:29 -05:00
fields = [
'device', 'name', 'type', 'description', 'tags',
2018-11-27 11:57:29 -05:00
]
2016-03-01 11:23:03 -05:00
widgets = {
'device': forms.HiddenInput(),
}
class ConsolePortCreateForm(ComponentForm):
2018-11-27 11:57:29 -05:00
name_pattern = ExpandableNameField(
label='Name'
)
type = forms.ChoiceField(
choices=CONSOLE_TYPE_CHOICES,
widget=StaticSelect2(),
)
description = forms.CharField(
max_length=100,
required=False
)
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
2016-03-01 11:23:03 -05:00
)
#
# Console server ports
#
class ConsoleServerPortForm(BootstrapMixin, forms.ModelForm):
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
model = ConsoleServerPort
2018-11-27 11:57:29 -05:00
fields = [
'device', 'name', 'type', 'description', 'tags',
2018-11-27 11:57:29 -05:00
]
2016-03-01 11:23:03 -05:00
widgets = {
'device': forms.HiddenInput(),
}
class ConsoleServerPortCreateForm(ComponentForm):
2018-11-27 11:57:29 -05:00
name_pattern = ExpandableNameField(
label='Name'
)
type = forms.ChoiceField(
choices=CONSOLE_TYPE_CHOICES,
widget=StaticSelect2(),
)
description = forms.CharField(
max_length=100,
required=False
)
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
2016-03-01 11:23:03 -05:00
)
class ConsoleServerPortBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
pk = forms.ModelMultipleChoiceField(
queryset=ConsoleServerPort.objects.all(),
widget=forms.MultipleHiddenInput()
)
type = forms.ChoiceField(
choices=add_blank_choice(CONSOLE_TYPE_CHOICES),
required=False,
widget=StaticSelect2()
)
description = forms.CharField(
max_length=100,
required=False
)
class Meta:
nullable_fields = [
'description',
]
class ConsoleServerPortBulkRenameForm(BulkRenameForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=ConsoleServerPort.objects.all(),
widget=forms.MultipleHiddenInput()
)
class ConsoleServerPortBulkDisconnectForm(ConfirmationForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=ConsoleServerPort.objects.all(),
widget=forms.MultipleHiddenInput()
)
2016-03-01 11:23:03 -05:00
#
# Power ports
#
class PowerPortForm(BootstrapMixin, forms.ModelForm):
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
model = PowerPort
2018-11-27 11:57:29 -05:00
fields = [
'device', 'name', 'maximum_draw', 'allocated_draw', 'description', 'tags',
2018-11-27 11:57:29 -05:00
]
2016-03-01 11:23:03 -05:00
widgets = {
'device': forms.HiddenInput(),
}
class PowerPortCreateForm(ComponentForm):
2018-11-27 11:57:29 -05:00
name_pattern = ExpandableNameField(
label='Name'
)
maximum_draw = forms.IntegerField(
min_value=1,
required=False,
help_text="Maximum draw in watts"
)
allocated_draw = forms.IntegerField(
min_value=1,
required=False,
help_text="Allocated draw in watts"
)
description = forms.CharField(
max_length=100,
required=False
)
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
2016-03-01 11:23:03 -05:00
)
#
# Power outlets
#
class PowerOutletForm(BootstrapMixin, forms.ModelForm):
power_port = forms.ModelChoiceField(
queryset=PowerPort.objects.all(),
required=False
)
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
model = PowerOutlet
2018-11-27 11:57:29 -05:00
fields = [
'device', 'name', 'power_port', 'feed_leg', 'description', 'tags',
2018-11-27 11:57:29 -05:00
]
2016-03-01 11:23:03 -05:00
widgets = {
'device': forms.HiddenInput(),
}
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# Limit power_port choices to the local device
if hasattr(self.instance, 'device'):
self.fields['power_port'].queryset = PowerPort.objects.filter(
device=self.instance.device
)
2016-03-01 11:23:03 -05:00
class PowerOutletCreateForm(ComponentForm):
2018-11-27 11:57:29 -05:00
name_pattern = ExpandableNameField(
label='Name'
)
power_port = forms.ModelChoiceField(
queryset=PowerPort.objects.all(),
required=False
)
feed_leg = forms.ChoiceField(
choices=add_blank_choice(POWERFEED_LEG_CHOICES),
required=False
)
description = forms.CharField(
max_length=100,
required=False
)
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
2016-03-01 11:23:03 -05:00
)
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# Limit power_port choices to those on the parent device
self.fields['power_port'].queryset = PowerPort.objects.filter(device=self.parent)
2016-03-01 11:23:03 -05:00
class PowerOutletBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
pk = forms.ModelMultipleChoiceField(
queryset=PowerOutlet.objects.all(),
widget=forms.MultipleHiddenInput()
)
feed_leg = forms.ChoiceField(
choices=add_blank_choice(POWERFEED_LEG_CHOICES),
required=False,
)
power_port = forms.ModelChoiceField(
queryset=PowerPort.objects.all(),
required=False
)
description = forms.CharField(
max_length=100,
required=False
)
class Meta:
nullable_fields = [
'feed_leg', 'power_port', 'description',
]
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# Limit power_port queryset to PowerPorts which belong to the parent Device
self.fields['power_port'].queryset = PowerPort.objects.filter(device=self.parent_obj)
class PowerOutletBulkRenameForm(BulkRenameForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=PowerOutlet.objects.all(),
widget=forms.MultipleHiddenInput
)
class PowerOutletBulkDisconnectForm(ConfirmationForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=PowerOutlet.objects.all(),
widget=forms.MultipleHiddenInput
)
2016-03-01 11:23:03 -05:00
#
# Interfaces
#
class InterfaceForm(InterfaceCommonForm, BootstrapMixin, forms.ModelForm):
untagged_vlan = forms.ModelChoiceField(
queryset=VLAN.objects.all(),
required=False,
widget=APISelect(
api_url="/api/ipam/vlans/",
display_field='display_name',
full=True
)
)
tagged_vlans = forms.ModelMultipleChoiceField(
queryset=VLAN.objects.all(),
required=False,
widget=APISelectMultiple(
api_url="/api/ipam/vlans/",
display_field='display_name',
full=True
)
)
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
2016-03-01 11:23:03 -05:00
class Meta:
model = Interface
fields = [
'device', 'name', 'type', 'enabled', 'lag', 'mac_address', 'mtu', 'mgmt_only', 'description',
'mode', 'untagged_vlan', 'tagged_vlans', 'tags',
]
2016-03-01 11:23:03 -05:00
widgets = {
'device': forms.HiddenInput(),
'type': StaticSelect2(),
'lag': StaticSelect2(),
'mode': StaticSelect2(),
2016-03-01 11:23:03 -05:00
}
labels = {
'mode': '802.1Q Mode',
}
help_texts = {
'mode': INTERFACE_MODE_HELP_TEXT,
}
2016-03-01 11:23:03 -05:00
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# Limit LAG choices to interfaces belonging to this device (or VC master)
if self.is_bound:
device = Device.objects.get(pk=self.data['device'])
self.fields['lag'].queryset = Interface.objects.filter(
device__in=[device, device.get_vc_master()], type=IFACE_TYPE_LAG
)
else:
device = self.instance.device
self.fields['lag'].queryset = Interface.objects.filter(
device__in=[self.instance.device, self.instance.device.get_vc_master()], type=IFACE_TYPE_LAG
)
# Limit VLan choices to those in: global vlans, global groups, the current site's group, the current site
vlan_choices = []
global_vlans = VLAN.objects.filter(site=None, group=None)
2019-01-09 23:33:08 -05:00
vlan_choices.append(
('Global', [(vlan.pk, vlan) for vlan in global_vlans])
)
for group in VLANGroup.objects.filter(site=None):
global_group_vlans = VLAN.objects.filter(group=group)
vlan_choices.append(
(group.name, [(vlan.pk, vlan) for vlan in global_group_vlans])
)
site = getattr(self.instance.parent, 'site', None)
if site is not None:
# Add non-grouped site VLANs
site_vlans = VLAN.objects.filter(site=site, group=None)
vlan_choices.append((site.name, [(vlan.pk, vlan) for vlan in site_vlans]))
# Add grouped site VLANs
for group in VLANGroup.objects.filter(site=site):
site_group_vlans = VLAN.objects.filter(group=group)
vlan_choices.append((
'{} / {}'.format(group.site.name, group.name),
[(vlan.pk, vlan) for vlan in site_group_vlans]
))
self.fields['untagged_vlan'].choices = [(None, '---------')] + vlan_choices
self.fields['tagged_vlans'].choices = vlan_choices
2018-03-08 13:25:51 -05:00
class InterfaceCreateForm(InterfaceCommonForm, ComponentForm, forms.Form):
2018-11-27 11:57:29 -05:00
name_pattern = ExpandableNameField(
label='Name'
)
type = forms.ChoiceField(
choices=IFACE_TYPE_CHOICES,
widget=StaticSelect2(),
2018-11-27 11:57:29 -05:00
)
enabled = forms.BooleanField(
required=False
)
lag = forms.ModelChoiceField(
queryset=Interface.objects.all(),
required=False,
label='Parent LAG',
widget=StaticSelect2(),
2018-11-27 11:57:29 -05:00
)
mtu = forms.IntegerField(
required=False,
min_value=1,
max_value=32767,
label='MTU'
)
mac_address = forms.CharField(
required=False,
label='MAC Address'
)
mgmt_only = forms.BooleanField(
required=False,
2018-11-27 11:57:29 -05:00
label='Management only',
help_text='This interface is used only for out-of-band management'
)
2018-11-27 11:57:29 -05:00
description = forms.CharField(
max_length=100,
required=False
)
mode = forms.ChoiceField(
choices=add_blank_choice(IFACE_MODE_CHOICES),
required=False,
widget=StaticSelect2(),
2018-11-27 11:57:29 -05:00
)
tags = TagField(
required=False
)
untagged_vlan = forms.ModelChoiceField(
queryset=VLAN.objects.all(),
required=False,
widget=APISelect(
api_url="/api/ipam/vlans/",
display_field='display_name',
full=True
)
)
tagged_vlans = forms.ModelMultipleChoiceField(
queryset=VLAN.objects.all(),
required=False,
widget=APISelectMultiple(
api_url="/api/ipam/vlans/",
display_field='display_name',
full=True
)
)
2016-03-01 11:23:03 -05: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)
# Limit LAG choices to interfaces belonging to this device (or its VC master)
if self.parent is not None:
self.fields['lag'].queryset = Interface.objects.filter(
device__in=[self.parent, self.parent.get_vc_master()], type=IFACE_TYPE_LAG
)
else:
self.fields['lag'].queryset = Interface.objects.none()
# Limit VLan choices to those in: global vlans, global groups, the current site's group, the current site
vlan_choices = []
global_vlans = VLAN.objects.filter(site=None, group=None)
vlan_choices.append(
('Global', [(vlan.pk, vlan) for vlan in global_vlans])
)
for group in VLANGroup.objects.filter(site=None):
global_group_vlans = VLAN.objects.filter(group=group)
vlan_choices.append(
(group.name, [(vlan.pk, vlan) for vlan in global_group_vlans])
)
site = getattr(self.parent, 'site', None)
if site is not None:
# Add non-grouped site VLANs
site_vlans = VLAN.objects.filter(site=site, group=None)
vlan_choices.append((site.name, [(vlan.pk, vlan) for vlan in site_vlans]))
# Add grouped site VLANs
for group in VLANGroup.objects.filter(site=site):
site_group_vlans = VLAN.objects.filter(group=group)
vlan_choices.append((
'{} / {}'.format(group.site.name, group.name),
[(vlan.pk, vlan) for vlan in site_group_vlans]
))
2016-03-01 11:23:03 -05:00
self.fields['untagged_vlan'].choices = [(None, '---------')] + vlan_choices
self.fields['tagged_vlans'].choices = vlan_choices
class InterfaceBulkEditForm(InterfaceCommonForm, BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=Interface.objects.all(),
widget=forms.MultipleHiddenInput()
)
type = forms.ChoiceField(
choices=add_blank_choice(IFACE_TYPE_CHOICES),
2019-01-08 15:35:34 -08:00
required=False,
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
enabled = forms.NullBooleanField(
required=False,
widget=BulkEditNullBooleanSelect()
)
lag = forms.ModelChoiceField(
queryset=Interface.objects.all(),
required=False,
2019-01-08 15:35:34 -08:00
label='Parent LAG',
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
mac_address = forms.CharField(
required=False,
label='MAC Address'
)
2018-11-27 11:57:29 -05:00
mtu = forms.IntegerField(
required=False,
min_value=1,
max_value=32767,
label='MTU'
)
mgmt_only = forms.NullBooleanField(
required=False,
widget=BulkEditNullBooleanSelect(),
label='Management only'
)
description = forms.CharField(
max_length=100,
required=False
)
mode = forms.ChoiceField(
choices=add_blank_choice(IFACE_MODE_CHOICES),
2019-01-08 15:35:34 -08:00
required=False,
widget=StaticSelect2()
2018-11-27 11:57:29 -05:00
)
untagged_vlan = forms.ModelChoiceField(
queryset=VLAN.objects.all(),
required=False,
widget=APISelect(
api_url="/api/ipam/vlans/",
display_field='display_name',
full=True
)
)
tagged_vlans = forms.ModelMultipleChoiceField(
queryset=VLAN.objects.all(),
required=False,
widget=APISelectMultiple(
api_url="/api/ipam/vlans/",
display_field='display_name',
full=True
)
)
2016-10-14 16:38:46 -04:00
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'lag', 'mac_address', 'mtu', 'description', 'mode', 'untagged_vlan', 'tagged_vlans'
2018-11-27 11:57:29 -05:00
]
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# Limit LAG choices to interfaces which belong to the parent device (or VC master)
device = self.parent_obj
if device is not None:
self.fields['lag'].queryset = Interface.objects.filter(
device__in=[device, device.get_vc_master()],
type=IFACE_TYPE_LAG
)
else:
self.fields['lag'].choices = []
2016-10-14 16:38:46 -04:00
# Limit VLan choices to those in: global vlans, global groups, the current site's group, the current site
vlan_choices = []
global_vlans = VLAN.objects.filter(site=None, group=None)
vlan_choices.append(
('Global', [(vlan.pk, vlan) for vlan in global_vlans])
)
for group in VLANGroup.objects.filter(site=None):
global_group_vlans = VLAN.objects.filter(group=group)
vlan_choices.append(
(group.name, [(vlan.pk, vlan) for vlan in global_group_vlans])
)
if self.parent_obj is not None:
site = getattr(self.parent_obj, 'site', None)
if site is not None:
# Add non-grouped site VLANs
site_vlans = VLAN.objects.filter(site=site, group=None)
vlan_choices.append((site.name, [(vlan.pk, vlan) for vlan in site_vlans]))
# Add grouped site VLANs
for group in VLANGroup.objects.filter(site=site):
site_group_vlans = VLAN.objects.filter(group=group)
vlan_choices.append((
'{} / {}'.format(group.site.name, group.name),
[(vlan.pk, vlan) for vlan in site_group_vlans]
))
self.fields['untagged_vlan'].choices = [(None, '---------')] + vlan_choices
self.fields['tagged_vlans'].choices = vlan_choices
2016-10-14 16:38:46 -04:00
class InterfaceBulkRenameForm(BulkRenameForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=Interface.objects.all(),
widget=forms.MultipleHiddenInput()
)
class InterfaceBulkDisconnectForm(ConfirmationForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=Interface.objects.all(),
widget=forms.MultipleHiddenInput()
)
2016-03-01 11:23:03 -05:00
#
# Front pass-through ports
2016-03-01 11:23:03 -05:00
#
class FrontPortForm(BootstrapMixin, forms.ModelForm):
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
2018-10-03 14:04:16 -04:00
class Meta:
model = FrontPort
2018-11-27 11:57:29 -05:00
fields = [
'device', 'name', 'type', 'rear_port', 'rear_port_position', 'description', 'tags',
]
2018-10-03 14:04:16 -04:00
widgets = {
'device': forms.HiddenInput(),
'type': StaticSelect2(),
'rear_port': StaticSelect2(),
2018-10-03 14:04:16 -04:00
}
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# Limit RearPort choices to the local device
if hasattr(self.instance, 'device'):
self.fields['rear_port'].queryset = self.fields['rear_port'].queryset.filter(
device=self.instance.device
)
2018-10-03 14:04:16 -04:00
# TODO: Merge with FrontPortTemplateCreateForm to remove duplicate logic
class FrontPortCreateForm(ComponentForm):
2018-10-03 14:04:16 -04:00
name_pattern = ExpandableNameField(
label='Name'
)
type = forms.ChoiceField(
choices=PORT_TYPE_CHOICES,
widget=StaticSelect2(),
2018-10-03 14:04:16 -04:00
)
rear_port_set = forms.MultipleChoiceField(
choices=[],
2018-10-03 14:04:16 -04:00
label='Rear ports',
2019-01-08 15:35:34 -08:00
help_text='Select one rear port assignment for each front port being created.',
2018-10-03 14:04:16 -04:00
)
description = forms.CharField(
required=False
)
2018-10-03 14:04:16 -04:00
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
2018-10-03 14:04:16 -04:00
# Determine which rear port positions are occupied. These will be excluded from the list of available mappings.
occupied_port_positions = [
(front_port.rear_port_id, front_port.rear_port_position)
2018-10-30 14:54:29 -04:00
for front_port in self.parent.frontports.all()
2018-10-03 14:04:16 -04:00
]
# Populate rear port choices
choices = []
rear_ports = RearPort.objects.filter(device=self.parent)
2018-10-03 14:04:16 -04:00
for rear_port in rear_ports:
for i in range(1, rear_port.positions + 1):
if (rear_port.pk, i) not in occupied_port_positions:
choices.append(
('{}:{}'.format(rear_port.pk, i), '{}:{}'.format(rear_port.name, i))
)
self.fields['rear_port_set'].choices = choices
def clean(self):
# Validate that the number of ports being created equals the number of selected (rear port, position) tuples
front_port_count = len(self.cleaned_data['name_pattern'])
rear_port_count = len(self.cleaned_data['rear_port_set'])
if front_port_count != rear_port_count:
raise forms.ValidationError({
'rear_port_set': 'The provided name pattern will create {} ports, however {} rear port assignments '
'were selected. These counts must match.'.format(front_port_count, rear_port_count)
})
def get_iterative_data(self, iteration):
# Assign rear port and position from selected set
rear_port, position = self.cleaned_data['rear_port_set'][iteration].split(':')
return {
'rear_port': int(rear_port),
'rear_port_position': int(position),
}
class FrontPortBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
pk = forms.ModelMultipleChoiceField(
queryset=FrontPort.objects.all(),
widget=forms.MultipleHiddenInput()
)
type = forms.ChoiceField(
choices=add_blank_choice(PORT_TYPE_CHOICES),
required=False,
widget=StaticSelect2()
)
description = forms.CharField(
max_length=100,
required=False
)
class Meta:
nullable_fields = [
'description',
]
class FrontPortBulkRenameForm(BulkRenameForm):
2018-10-03 14:04:16 -04:00
pk = forms.ModelMultipleChoiceField(
queryset=FrontPort.objects.all(),
2018-10-03 14:04:16 -04:00
widget=forms.MultipleHiddenInput
)
class FrontPortBulkDisconnectForm(ConfirmationForm):
pk = forms.ModelMultipleChoiceField(
queryset=FrontPort.objects.all(),
widget=forms.MultipleHiddenInput
)
2018-10-03 14:04:16 -04:00
#
# Rear pass-through ports
2018-10-03 14:04:16 -04:00
#
class RearPortForm(BootstrapMixin, forms.ModelForm):
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
2018-10-03 14:04:16 -04:00
class Meta:
model = RearPort
2018-11-27 11:57:29 -05:00
fields = [
'device', 'name', 'type', 'positions', 'description', 'tags',
]
2018-10-03 14:04:16 -04:00
widgets = {
'device': forms.HiddenInput(),
'type': StaticSelect2(),
2018-10-03 14:04:16 -04:00
}
class RearPortCreateForm(ComponentForm):
2018-10-03 14:04:16 -04:00
name_pattern = ExpandableNameField(
label='Name'
)
type = forms.ChoiceField(
choices=PORT_TYPE_CHOICES,
widget=StaticSelect2(),
2018-10-03 14:04:16 -04:00
)
positions = forms.IntegerField(
min_value=1,
max_value=64,
initial=1,
help_text='The number of front ports which may be mapped to each rear port'
)
description = forms.CharField(
required=False
)
2018-10-03 14:04:16 -04:00
class RearPortBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
pk = forms.ModelMultipleChoiceField(
queryset=RearPort.objects.all(),
widget=forms.MultipleHiddenInput()
)
type = forms.ChoiceField(
choices=add_blank_choice(PORT_TYPE_CHOICES),
required=False,
widget=StaticSelect2()
)
description = forms.CharField(
max_length=100,
required=False
)
class Meta:
nullable_fields = [
'description',
]
class RearPortBulkRenameForm(BulkRenameForm):
2018-10-03 14:04:16 -04:00
pk = forms.ModelMultipleChoiceField(
queryset=RearPort.objects.all(),
2018-10-03 14:04:16 -04:00
widget=forms.MultipleHiddenInput
)
class RearPortBulkDisconnectForm(ConfirmationForm):
pk = forms.ModelMultipleChoiceField(
queryset=RearPort.objects.all(),
widget=forms.MultipleHiddenInput
)
#
# Cables
#
2019-03-21 17:47:43 -04:00
class ConnectCableToDeviceForm(BootstrapMixin, ChainedFieldsMixin, forms.ModelForm):
"""
Base form for connecting a Cable to a Device component
"""
termination_b_site = forms.ModelChoiceField(
queryset=Site.objects.all(),
label='Site',
required=False,
2019-01-03 03:00:27 -05:00
widget=APISelect(
api_url='/api/dcim/sites/',
filter_for={
'termination_b_rack': 'site_id',
'termination_b_device': 'site_id',
2018-11-27 11:57:29 -05:00
}
)
)
termination_b_rack = ChainedModelChoiceField(
queryset=Rack.objects.all(),
chains=(
('site', 'termination_b_site'),
),
label='Rack',
required=False,
widget=APISelect(
2018-12-30 02:35:18 -05:00
api_url='/api/dcim/racks/',
2019-01-03 03:00:27 -05:00
filter_for={
'termination_b_device': 'rack_id',
},
2018-11-27 11:57:29 -05:00
attrs={
'nullable': 'true',
}
)
)
termination_b_device = ChainedModelChoiceField(
queryset=Device.objects.all(),
chains=(
('site', 'termination_b_site'),
('rack', 'termination_b_rack'),
),
label='Device',
2019-01-03 03:00:27 -05:00
required=False,
widget=APISelect(
2018-12-30 02:35:18 -05:00
api_url='/api/dcim/devices/',
display_field='display_name',
2019-01-03 03:00:27 -05:00
filter_for={
'termination_b_id': 'device_id',
2018-11-27 11:57:29 -05:00
}
)
)
class Meta:
model = Cable
fields = [
'termination_b_site', 'termination_b_rack', 'termination_b_device', 'termination_b_id', 'type', 'status',
'label', 'color', 'length', 'length_unit',
]
class ConnectCableToConsolePortForm(ConnectCableToDeviceForm):
termination_b_id = forms.IntegerField(
label='Name',
widget=APISelect(
api_url='/api/dcim/console-ports/',
disabled_indicator='cable',
)
)
class ConnectCableToConsoleServerPortForm(ConnectCableToDeviceForm):
termination_b_id = forms.IntegerField(
label='Name',
widget=APISelect(
api_url='/api/dcim/console-server-ports/',
disabled_indicator='cable',
)
)
class ConnectCableToPowerPortForm(ConnectCableToDeviceForm):
termination_b_id = forms.IntegerField(
label='Name',
widget=APISelect(
api_url='/api/dcim/power-ports/',
disabled_indicator='cable',
)
)
class ConnectCableToPowerOutletForm(ConnectCableToDeviceForm):
termination_b_id = forms.IntegerField(
label='Name',
widget=APISelect(
api_url='/api/dcim/power-outlets/',
disabled_indicator='cable',
)
)
class ConnectCableToInterfaceForm(ConnectCableToDeviceForm):
termination_b_id = forms.IntegerField(
label='Name',
widget=APISelect(
api_url='/api/dcim/interfaces/',
disabled_indicator='cable',
additional_query_params={
'kind': 'physical',
}
)
2016-03-01 11:23:03 -05:00
)
class ConnectCableToFrontPortForm(ConnectCableToDeviceForm):
termination_b_id = forms.IntegerField(
label='Name',
widget=APISelect(
api_url='/api/dcim/front-ports/',
disabled_indicator='cable',
)
)
class ConnectCableToRearPortForm(ConnectCableToDeviceForm):
termination_b_id = forms.IntegerField(
label='Name',
widget=APISelect(
api_url='/api/dcim/rear-ports/',
disabled_indicator='cable',
)
)
2016-03-01 11:23:03 -05:00
class ConnectCableToCircuitTerminationForm(BootstrapMixin, ChainedFieldsMixin, forms.ModelForm):
2019-03-21 17:47:43 -04:00
termination_b_provider = forms.ModelChoiceField(
queryset=Provider.objects.all(),
label='Provider',
widget=APISelect(
api_url='/api/circuits/providers/',
filter_for={
'termination_b_circuit': 'provider_id',
}
)
)
termination_b_site = forms.ModelChoiceField(
queryset=Site.objects.all(),
label='Site',
required=False,
widget=APISelect(
api_url='/api/dcim/sites/',
filter_for={
'termination_b_circuit': 'site_id',
}
)
)
2019-03-21 17:47:43 -04:00
termination_b_circuit = ChainedModelChoiceField(
queryset=Circuit.objects.all(),
chains=(
('provider', 'termination_b_provider'),
),
label='Circuit',
widget=APISelect(
api_url='/api/circuits/circuits/',
display_field='cid',
filter_for={
'termination_b_id': 'circuit_id',
}
)
)
termination_b_id = forms.IntegerField(
label='Side',
2019-03-21 17:47:43 -04:00
widget=APISelect(
api_url='/api/circuits/circuit-terminations/',
disabled_indicator='cable',
display_field='term_side'
2019-03-21 17:47:43 -04:00
)
)
class Meta:
model = Cable
fields = [
'termination_b_provider', 'termination_b_site', 'termination_b_circuit', 'termination_b_id', 'type',
'status', 'label', 'color', 'length', 'length_unit',
2019-03-21 17:47:43 -04:00
]
class ConnectCableToPowerFeedForm(BootstrapMixin, ChainedFieldsMixin, forms.ModelForm):
termination_b_site = forms.ModelChoiceField(
queryset=Site.objects.all(),
label='Site',
widget=APISelect(
api_url='/api/dcim/sites/',
display_field='cid',
filter_for={
'termination_b_rackgroup': 'site_id',
'termination_b_powerpanel': 'site_id',
}
)
)
termination_b_rackgroup = ChainedModelChoiceField(
queryset=RackGroup.objects.all(),
label='Rack Group',
chains=(
('site', 'termination_b_site'),
),
required=False,
widget=APISelect(
api_url='/api/dcim/rack-groups/',
display_field='cid',
filter_for={
'termination_b_powerpanel': 'rackgroup_id',
}
)
)
termination_b_powerpanel = ChainedModelChoiceField(
queryset=PowerPanel.objects.all(),
chains=(
('site', 'termination_b_site'),
('rack_group', 'termination_b_rackgroup'),
),
label='Power Panel',
widget=APISelect(
api_url='/api/dcim/power-panels/',
filter_for={
'termination_b_id': 'power_panel_id',
2019-03-21 17:47:43 -04:00
}
)
)
termination_b_id = forms.IntegerField(
label='Name',
2019-03-21 17:47:43 -04:00
widget=APISelect(
api_url='/api/dcim/power-feeds/',
)
)
class Meta:
model = Cable
fields = [
'termination_b_rackgroup', 'termination_b_powerpanel', 'termination_b_id', 'type', 'status', 'label',
'color', 'length', 'length_unit',
]
2018-10-25 15:51:12 -04:00
class CableForm(BootstrapMixin, forms.ModelForm):
class Meta:
model = Cable
2018-11-27 11:57:29 -05:00
fields = [
'type', 'status', 'label', 'color', 'length', 'length_unit',
2016-03-01 11:23:03 -05:00
]
2018-10-31 17:05:25 -04:00
class CableCSVForm(forms.ModelForm):
2016-03-01 11:23:03 -05:00
2018-10-31 17:05:25 -04:00
# Termination A
side_a_device = FlexibleModelChoiceField(
queryset=Device.objects.all(),
to_field_name='name',
2018-11-01 09:59:53 -04:00
help_text='Side A device name or ID',
2018-10-31 17:05:25 -04:00
error_messages={
'invalid_choice': 'Side A device not found',
}
)
side_a_type = forms.ModelChoiceField(
queryset=ContentType.objects.all(),
2018-11-27 11:57:29 -05:00
limit_choices_to={
'model__in': CABLE_TERMINATION_TYPES,
},
2018-11-01 09:59:53 -04:00
to_field_name='model',
help_text='Side A type'
)
2018-11-01 09:59:53 -04:00
side_a_name = forms.CharField(
help_text='Side A component'
)
2018-10-31 17:05:25 -04:00
# Termination B
side_b_device = FlexibleModelChoiceField(
queryset=Device.objects.all(),
to_field_name='name',
2018-11-01 09:59:53 -04:00
help_text='Side B device name or ID',
2018-10-31 17:05:25 -04:00
error_messages={
'invalid_choice': 'Side B device not found',
}
)
side_b_type = forms.ModelChoiceField(
queryset=ContentType.objects.all(),
2018-11-27 11:57:29 -05:00
limit_choices_to={
'model__in': CABLE_TERMINATION_TYPES,
},
2018-11-01 09:59:53 -04:00
to_field_name='model',
help_text='Side B type'
)
2018-11-01 09:59:53 -04:00
side_b_name = forms.CharField(
help_text='Side B component'
)
2018-10-31 17:05:25 -04:00
# Cable attributes
status = CSVChoiceField(
choices=CONNECTION_STATUS_CHOICES,
2018-10-31 17:05:25 -04:00
required=False,
help_text='Connection status'
)
2018-11-01 09:59:53 -04:00
type = CSVChoiceField(
choices=CABLE_TYPE_CHOICES,
required=False,
help_text='Cable type'
)
length_unit = CSVChoiceField(
choices=CABLE_LENGTH_UNIT_CHOICES,
2018-11-01 09:59:53 -04:00
required=False,
help_text='Length unit'
)
2016-03-01 11:23:03 -05:00
class Meta:
2018-10-31 17:05:25 -04:00
model = Cable
fields = [
2018-11-01 09:59:53 -04:00
'side_a_device', 'side_a_type', 'side_a_name', 'side_b_device', 'side_b_type', 'side_b_name', 'type',
'status', 'label', 'color', 'length', 'length_unit',
2018-10-31 17:05:25 -04:00
]
2018-11-01 09:59:53 -04:00
help_texts = {
'color': 'RGB color in hexadecimal (e.g. 00ff00)'
}
2016-03-01 11:23:03 -05:00
2018-10-31 17:05:25 -04:00
# TODO: Merge the clean() methods for either end
def clean_side_a_name(self):
2016-03-01 11:23:03 -05:00
2018-10-31 17:05:25 -04:00
device = self.cleaned_data.get('side_a_device')
content_type = self.cleaned_data.get('side_a_type')
name = self.cleaned_data.get('side_a_name')
if not device or not content_type or not name:
return None
2016-03-01 11:23:03 -05:00
2018-10-31 17:05:25 -04:00
model = content_type.model_class()
try:
2018-10-31 17:05:25 -04:00
termination_object = model.objects.get(
device=device,
name=name
)
if termination_object.cable is not None:
raise forms.ValidationError(
"Side A: {} {} is already connected".format(device, termination_object)
)
except ObjectDoesNotExist:
raise forms.ValidationError(
"A side termination not found: {} {}".format(device, name)
)
2016-03-01 11:23:03 -05:00
2018-10-31 17:05:25 -04:00
self.instance.termination_a = termination_object
return termination_object
2016-03-01 11:23:03 -05:00
2018-10-31 17:05:25 -04:00
def clean_side_b_name(self):
2016-03-01 11:23:03 -05:00
2018-10-31 17:05:25 -04:00
device = self.cleaned_data.get('side_b_device')
content_type = self.cleaned_data.get('side_b_type')
name = self.cleaned_data.get('side_b_name')
if not device or not content_type or not name:
return None
2016-03-01 11:23:03 -05:00
2018-10-31 17:05:25 -04:00
model = content_type.model_class()
try:
2018-10-31 17:05:25 -04:00
termination_object = model.objects.get(
device=device,
name=name
)
2018-10-31 17:05:25 -04:00
if termination_object.cable is not None:
raise forms.ValidationError(
"Side B: {} {} is already connected".format(device, termination_object)
)
except ObjectDoesNotExist:
raise forms.ValidationError(
"B side termination not found: {} {}".format(device, name)
)
self.instance.termination_b = termination_object
return termination_object
def clean_length_unit(self):
# Avoid trying to save as NULL
length_unit = self.cleaned_data.get('length_unit', None)
return length_unit if length_unit is not None else ''
2018-10-31 17:05:25 -04:00
class CableBulkEditForm(BootstrapMixin, BulkEditForm):
pk = forms.ModelMultipleChoiceField(
queryset=Cable.objects.all(),
widget=forms.MultipleHiddenInput
)
type = forms.ChoiceField(
choices=add_blank_choice(CABLE_TYPE_CHOICES),
required=False,
2019-01-08 15:35:34 -08:00
initial='',
widget=StaticSelect2()
)
status = forms.ChoiceField(
choices=add_blank_choice(CONNECTION_STATUS_CHOICES),
required=False,
widget=StaticSelect2(),
initial=''
)
label = forms.CharField(
max_length=100,
required=False
)
color = forms.CharField(
max_length=6,
required=False,
widget=ColorSelect()
)
length = forms.IntegerField(
min_value=1,
required=False
)
length_unit = forms.ChoiceField(
choices=add_blank_choice(CABLE_LENGTH_UNIT_CHOICES),
required=False,
2019-01-08 15:35:34 -08:00
initial='',
widget=StaticSelect2()
)
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'type', 'status', 'label', 'color', 'length',
]
def clean(self):
# Validate length/unit
length = self.cleaned_data.get('length')
length_unit = self.cleaned_data.get('length_unit')
if length and not length_unit:
raise forms.ValidationError({
'length_unit': "Must specify a unit when setting length"
})
2016-03-01 11:23:03 -05:00
2018-10-24 14:59:46 -04:00
class CableFilterForm(BootstrapMixin, forms.Form):
model = Cable
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
site = FilterChoiceField(
queryset=Site.objects.all(),
to_field_name='slug',
widget=APISelectMultiple(
api_url="/api/dcim/sites/",
value_field="slug",
filter_for={
'rack_id': 'site',
}
)
)
rack_id = FilterChoiceField(
queryset=Rack.objects.all(),
label='Rack',
null_label='-- None --',
widget=APISelectMultiple(
api_url="/api/dcim/racks/",
null_option=True,
)
)
2019-01-08 15:35:34 -08:00
type = forms.MultipleChoiceField(
choices=add_blank_choice(CABLE_TYPE_CHOICES),
2019-01-08 15:35:34 -08:00
required=False,
widget=StaticSelect2()
2018-10-24 14:59:46 -04:00
)
status = forms.ChoiceField(
required=False,
choices=add_blank_choice(CONNECTION_STATUS_CHOICES),
widget=StaticSelect2()
)
2019-01-08 15:35:34 -08:00
color = forms.CharField(
max_length=6,
required=False,
widget=ColorSelect()
2018-10-24 14:59:46 -04:00
)
device = forms.CharField(
required=False,
label='Device name'
)
2016-03-01 11:23:03 -05:00
#
# Device bays
#
class DeviceBayForm(BootstrapMixin, forms.ModelForm):
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
class Meta:
model = DeviceBay
2018-11-27 11:57:29 -05:00
fields = [
'device', 'name', 'description', 'tags',
2018-11-27 11:57:29 -05:00
]
widgets = {
'device': forms.HiddenInput(),
}
class DeviceBayCreateForm(ComponentForm):
2018-11-27 11:57:29 -05:00
name_pattern = ExpandableNameField(
label='Name'
)
tags = TagField(
required=False
)
class PopulateDeviceBayForm(BootstrapMixin, forms.Form):
installed_device = forms.ModelChoiceField(
queryset=Device.objects.all(),
label='Child Device',
help_text="Child devices must first be created and assigned to the site/rack of the parent device.",
widget=StaticSelect2(),
)
def __init__(self, device_bay, *args, **kwargs):
super().__init__(*args, **kwargs)
self.fields['installed_device'].queryset = Device.objects.filter(
site=device_bay.device.site,
rack=device_bay.device.rack,
parent_bay__isnull=True,
device_type__u_height=0,
device_type__subdevice_role=SUBDEVICE_ROLE_CHILD
).exclude(pk=device_bay.device.pk)
class DeviceBayBulkRenameForm(BulkRenameForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=DeviceBay.objects.all(),
widget=forms.MultipleHiddenInput()
)
2016-03-01 11:23:03 -05:00
#
# Connections
#
class ConsoleConnectionFilterForm(BootstrapMixin, forms.Form):
2018-11-27 11:57:29 -05:00
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
required=False,
to_field_name='slug'
)
device = forms.CharField(
required=False,
label='Device name'
)
2016-03-01 11:23:03 -05:00
class PowerConnectionFilterForm(BootstrapMixin, forms.Form):
2018-11-27 11:57:29 -05:00
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
required=False,
to_field_name='slug'
)
device = forms.CharField(
required=False,
label='Device name'
)
2016-03-01 11:23:03 -05:00
class InterfaceConnectionFilterForm(BootstrapMixin, forms.Form):
2018-11-27 11:57:29 -05:00
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
required=False,
to_field_name='slug'
)
device = forms.CharField(
required=False,
label='Device name'
)
2016-03-01 11:23:03 -05:00
#
# Inventory items
#
class InventoryItemForm(BootstrapMixin, forms.ModelForm):
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
class Meta:
model = InventoryItem
2018-11-27 11:57:29 -05:00
fields = [
'name', 'manufacturer', 'part_id', 'serial', 'asset_tag', 'description', 'tags',
]
widgets = {
'manufacturer': APISelect(
api_url="/api/dcim/manufacturers/"
)
}
class InventoryItemCSVForm(forms.ModelForm):
device = FlexibleModelChoiceField(
queryset=Device.objects.all(),
to_field_name='name',
help_text='Device name or ID',
error_messages={
'invalid_choice': 'Device not found.',
}
)
manufacturer = forms.ModelChoiceField(
queryset=Manufacturer.objects.all(),
to_field_name='name',
required=False,
help_text='Manufacturer name',
error_messages={
'invalid_choice': 'Invalid manufacturer.',
}
)
class Meta:
model = InventoryItem
fields = InventoryItem.csv_headers
class InventoryItemBulkEditForm(BootstrapMixin, BulkEditForm):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=InventoryItem.objects.all(),
widget=forms.MultipleHiddenInput()
)
manufacturer = forms.ModelChoiceField(
queryset=Manufacturer.objects.all(),
required=False
)
part_id = forms.CharField(
max_length=50,
required=False,
label='Part ID'
)
description = forms.CharField(
max_length=100,
required=False
)
class Meta:
2018-11-27 11:57:29 -05:00
nullable_fields = [
'manufacturer', 'part_id', 'description',
]
class InventoryItemFilterForm(BootstrapMixin, forms.Form):
model = InventoryItem
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
device = forms.CharField(
required=False,
label='Device name'
)
manufacturer = FilterChoiceField(
2019-01-09 13:44:39 -05:00
queryset=Manufacturer.objects.all(),
to_field_name='slug',
null_label='-- None --'
)
discovered = forms.NullBooleanField(
required=False,
widget=forms.Select(
choices=BOOLEAN_WITH_BLANK_CHOICES
)
)
#
# Virtual chassis
#
class DeviceSelectionForm(forms.Form):
2018-11-27 11:57:29 -05:00
pk = forms.ModelMultipleChoiceField(
queryset=Device.objects.all(),
widget=forms.MultipleHiddenInput()
)
class VirtualChassisForm(BootstrapMixin, forms.ModelForm):
2018-11-27 11:57:29 -05:00
tags = TagField(
required=False
)
class Meta:
model = VirtualChassis
2018-11-27 11:57:29 -05:00
fields = [
'master', 'domain', 'tags',
]
widgets = {
2018-11-27 11:57:29 -05:00
'master': SelectWithPK(),
}
2018-02-14 12:05:00 -05:00
class BaseVCMemberFormSet(forms.BaseModelFormSet):
def clean(self):
super().clean()
2018-02-14 12:05:00 -05:00
# Check for duplicate VC position values
vc_position_list = []
for form in self.forms:
vc_position = form.cleaned_data.get('vc_position')
if vc_position:
if vc_position in vc_position_list:
error_msg = 'A virtual chassis member already exists in position {}.'.format(vc_position)
form.add_error('vc_position', error_msg)
vc_position_list.append(vc_position)
2018-02-14 12:05:00 -05:00
class DeviceVCMembershipForm(forms.ModelForm):
class Meta:
model = Device
2018-11-27 11:57:29 -05:00
fields = [
'vc_position', 'vc_priority',
]
2018-02-14 12:05:00 -05:00
labels = {
'vc_position': 'Position',
'vc_priority': 'Priority',
}
def __init__(self, validate_vc_position=False, *args, **kwargs):
super().__init__(*args, **kwargs)
2018-02-14 12:05:00 -05:00
# Require VC position (only required when the Device is a VirtualChassis member)
self.fields['vc_position'].required = True
# Validation of vc_position is optional. This is only required when adding a new member to an existing
# VirtualChassis. Otherwise, vc_position validation is handled by BaseVCMemberFormSet.
self.validate_vc_position = validate_vc_position
def clean_vc_position(self):
vc_position = self.cleaned_data['vc_position']
if self.validate_vc_position:
conflicting_members = Device.objects.filter(
virtual_chassis=self.instance.virtual_chassis,
vc_position=vc_position
)
if conflicting_members.exists():
raise forms.ValidationError(
'A virtual chassis member already exists in position {}.'.format(vc_position)
)
return vc_position
2018-02-01 11:39:13 -05:00
class VCMemberSelectForm(BootstrapMixin, ChainedFieldsMixin, forms.Form):
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
label='Site',
required=False,
widget=APISelect(
api_url="/api/dcim/sites/",
filter_for={
'rack': 'site_id',
'device': 'site_id',
2018-11-27 11:57:29 -05:00
}
2018-02-01 11:39:13 -05:00
)
)
rack = ChainedModelChoiceField(
queryset=Rack.objects.all(),
chains=(
('site', 'site'),
),
label='Rack',
required=False,
widget=APISelect(
api_url='/api/dcim/racks/',
filter_for={
'device': 'rack_id'
},
2018-11-27 11:57:29 -05:00
attrs={
'nullable': 'true',
}
2018-02-01 11:39:13 -05:00
)
)
device = ChainedModelChoiceField(
2018-11-27 11:57:29 -05:00
queryset=Device.objects.filter(
virtual_chassis__isnull=True
),
2018-02-01 11:39:13 -05:00
chains=(
('site', 'site'),
('rack', 'rack'),
),
label='Device',
widget=APISelect(
api_url='/api/dcim/devices/',
display_field='display_name',
disabled_indicator='virtual_chassis'
2018-02-01 11:39:13 -05:00
)
)
2018-02-01 15:53:59 -05:00
def clean_device(self):
device = self.cleaned_data['device']
if device.virtual_chassis is not None:
2018-11-27 11:57:29 -05:00
raise forms.ValidationError(
"Device {} is already assigned to a virtual chassis.".format(device)
)
2018-02-14 12:05:00 -05:00
return device
2018-02-21 09:53:23 -05:00
class VirtualChassisFilterForm(BootstrapMixin, CustomFieldFilterForm):
model = VirtualChassis
2018-11-27 11:57:29 -05:00
q = forms.CharField(
required=False,
label='Search'
)
2018-02-21 09:53:23 -05:00
site = FilterChoiceField(
queryset=Site.objects.all(),
to_field_name='slug',
2019-05-09 14:32:49 -04:00
widget=APISelectMultiple(
api_url="/api/dcim/sites/",
value_field="slug",
)
)
tenant_group = FilterChoiceField(
queryset=TenantGroup.objects.all(),
to_field_name='slug',
null_label='-- None --',
widget=APISelectMultiple(
api_url="/api/tenancy/tenant-groups/",
value_field="slug",
null_option=True,
filter_for={
'tenant': 'group'
}
)
2018-02-21 09:53:23 -05:00
)
tenant = FilterChoiceField(
queryset=Tenant.objects.all(),
to_field_name='slug',
null_label='-- None --',
2019-05-09 14:32:49 -04:00
widget=APISelectMultiple(
api_url="/api/tenancy/tenants/",
value_field="slug",
null_option=True,
)
2018-02-21 09:53:23 -05:00
)
2019-03-11 22:40:52 -04:00
#
# Power panels
#
class PowerPanelForm(BootstrapMixin, forms.ModelForm):
2019-03-12 10:15:56 -04:00
rack_group = ChainedModelChoiceField(
2019-03-11 22:40:52 -04:00
queryset=RackGroup.objects.all(),
chains=(
('site', 'site'),
),
required=False,
widget=APISelect(
api_url='/api/dcim/rack-groups/',
)
)
class Meta:
model = PowerPanel
fields = [
2019-03-12 10:15:56 -04:00
'site', 'rack_group', 'name',
2019-03-11 22:40:52 -04:00
]
widgets = {
'site': APISelect(
api_url="/api/dcim/sites/",
filter_for={
2019-03-12 11:36:29 -04:00
'rack_group': 'site_id',
2019-03-11 22:40:52 -04:00
}
),
}
class PowerPanelCSVForm(forms.ModelForm):
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
to_field_name='name',
help_text='Name of parent site',
error_messages={
'invalid_choice': 'Site not found.',
}
)
2019-04-09 14:55:17 -04:00
rack_group_name = forms.CharField(
required=False,
help_text="Rack group name (optional)"
2019-03-11 22:40:52 -04:00
)
class Meta:
model = PowerPanel
fields = PowerPanel.csv_headers
2019-04-09 14:55:17 -04:00
def clean(self):
super().clean()
site = self.cleaned_data.get('site')
rack_group_name = self.cleaned_data.get('rack_group_name')
# Validate rack group
if rack_group_name:
try:
self.instance.rack_group = RackGroup.objects.get(site=site, name=rack_group_name)
except RackGroup.DoesNotExist:
raise forms.ValidationError(
"Rack group {} not found in site {}".format(rack_group_name, site)
)
2019-03-11 22:40:52 -04:00
2019-03-12 12:05:58 -04:00
class PowerPanelFilterForm(BootstrapMixin, CustomFieldFilterForm):
model = PowerPanel
q = forms.CharField(
required=False,
label='Search'
)
site = FilterChoiceField(
queryset=Site.objects.all(),
to_field_name='slug',
widget=APISelectMultiple(
api_url="/api/dcim/sites/",
value_field="slug",
filter_for={
2019-04-19 13:32:46 -04:00
'rack_group_id': 'site',
2019-03-12 12:05:58 -04:00
}
)
)
rack_group_id = FilterChoiceField(
queryset=RackGroup.objects.all(),
label='Rack group (ID)',
null_label='-- None --',
widget=APISelectMultiple(
api_url="/api/dcim/rack-groups/",
null_option=True,
)
)
2019-03-11 22:40:52 -04:00
#
# Power feeds
#
class PowerFeedForm(BootstrapMixin, CustomFieldForm):
2019-03-12 10:15:56 -04:00
site = ChainedModelChoiceField(
queryset=Site.objects.all(),
required=False,
widget=APISelect(
api_url='/api/dcim/sites/',
filter_for={
'power_panel': 'site_id',
'rack': 'site_id',
}
)
)
2019-04-11 10:49:43 -04:00
comments = CommentField()
2019-03-11 22:40:52 -04:00
tags = TagField(
required=False
)
class Meta:
model = PowerFeed
fields = [
2019-03-12 11:36:29 -04:00
'site', 'power_panel', 'rack', 'name', 'status', 'type', 'supply', 'phase', 'voltage', 'amperage',
2019-06-17 14:52:11 -04:00
'max_utilization', 'comments', 'tags',
2019-03-11 22:40:52 -04:00
]
widgets = {
2019-03-12 10:15:56 -04:00
'power_panel': APISelect(
api_url="/api/dcim/power-panels/"
),
'rack': APISelect(
api_url="/api/dcim/racks/"
2019-03-11 22:40:52 -04:00
),
'status': StaticSelect2(),
2019-03-12 11:36:29 -04:00
'type': StaticSelect2(),
2019-03-11 22:40:52 -04:00
'supply': StaticSelect2(),
'phase': StaticSelect2(),
}
2019-04-11 10:49:43 -04:00
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# Initialize site field
if self.instance and hasattr(self.instance, 'power_panel'):
2019-04-11 10:49:43 -04:00
self.initial['site'] = self.instance.power_panel.site
2019-03-11 22:40:52 -04:00
class PowerFeedCSVForm(forms.ModelForm):
2019-04-09 14:55:17 -04:00
site = forms.ModelChoiceField(
queryset=Site.objects.all(),
to_field_name='name',
help_text='Name of parent site',
error_messages={
'invalid_choice': 'Site not found.',
}
)
panel_name = forms.ModelChoiceField(
queryset=PowerPanel.objects.all(),
to_field_name='name',
help_text='Name of upstream power panel',
error_messages={
'invalid_choice': 'Power panel not found.',
}
)
rack_group = forms.CharField(
required=False,
help_text="Rack group name (optional)"
)
rack_name = forms.CharField(
required=False,
help_text="Rack name (optional)"
)
2019-03-11 22:40:52 -04:00
status = CSVChoiceField(
choices=POWERFEED_STATUS_CHOICES,
required=False,
help_text='Operational status'
)
2019-03-12 11:36:29 -04:00
type = CSVChoiceField(
choices=POWERFEED_TYPE_CHOICES,
required=False,
help_text='Primary or redundant'
)
2019-03-11 22:40:52 -04:00
supply = CSVChoiceField(
choices=POWERFEED_SUPPLY_CHOICES,
required=False,
help_text='AC/DC'
)
2019-04-09 14:55:17 -04:00
phase = CSVChoiceField(
choices=POWERFEED_PHASE_CHOICES,
required=False,
help_text='Single or three-phase'
)
2019-03-11 22:40:52 -04:00
class Meta:
model = PowerFeed
fields = PowerFeed.csv_headers
2019-04-09 14:55:17 -04:00
def clean(self):
super().clean()
site = self.cleaned_data.get('site')
panel_name = self.cleaned_data.get('panel_name')
rack_group = self.cleaned_data.get('rack_group')
rack_name = self.cleaned_data.get('rack_name')
# Validate power panel
if panel_name:
try:
self.instance.power_panel = PowerPanel.objects.get(site=site, name=panel_name)
except Rack.DoesNotExist:
raise forms.ValidationError(
"Power panel {} not found in site {}".format(panel_name, site)
)
# Validate rack
if rack_name:
try:
self.instance.rack = Rack.objects.get(site=site, group__name=rack_group, name=rack_name)
2019-04-09 14:55:17 -04:00
except Rack.DoesNotExist:
raise forms.ValidationError(
"Rack {} not found in site {}, group {}".format(rack_name, site, rack_group)
)
2019-03-11 22:40:52 -04:00
class PowerFeedBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
pk = forms.ModelMultipleChoiceField(
queryset=PowerFeed.objects.all(),
widget=forms.MultipleHiddenInput
)
powerpanel = forms.ModelChoiceField(
queryset=PowerPanel.objects.all(),
required=False,
widget=APISelect(
api_url="/api/dcim/power-panels/",
2019-03-11 22:40:52 -04:00
filter_for={
'rackgroup': 'site_id',
}
)
)
2019-04-11 10:49:43 -04:00
rack = forms.ModelChoiceField(
queryset=Rack.objects.all(),
2019-03-11 22:40:52 -04:00
required=False,
widget=APISelect(
2019-04-11 10:49:43 -04:00
api_url="/api/dcim/racks",
2019-03-11 22:40:52 -04:00
)
)
2019-03-12 11:36:29 -04:00
status = forms.ChoiceField(
choices=add_blank_choice(POWERFEED_STATUS_CHOICES),
2019-03-11 22:40:52 -04:00
required=False,
initial='',
widget=StaticSelect2()
)
2019-03-12 11:36:29 -04:00
type = forms.ChoiceField(
choices=add_blank_choice(POWERFEED_TYPE_CHOICES),
2019-03-11 22:40:52 -04:00
required=False,
initial='',
widget=StaticSelect2()
)
supply = forms.ChoiceField(
choices=add_blank_choice(POWERFEED_SUPPLY_CHOICES),
required=False,
initial='',
widget=StaticSelect2()
)
phase = forms.ChoiceField(
choices=add_blank_choice(POWERFEED_PHASE_CHOICES),
required=False,
initial='',
widget=StaticSelect2()
)
2019-03-12 11:36:29 -04:00
voltage = forms.IntegerField(
required=False
)
amperage = forms.IntegerField(
required=False
)
2019-06-17 14:52:11 -04:00
max_utilization = forms.IntegerField(
2019-03-11 22:40:52 -04:00
required=False
)
comments = forms.CharField(
required=False
)
class Meta:
nullable_fields = [
'rackgroup', 'comments',
]
2019-03-12 12:05:58 -04:00
class PowerFeedFilterForm(BootstrapMixin, CustomFieldFilterForm):
model = PowerFeed
q = forms.CharField(
required=False,
label='Search'
)
site = FilterChoiceField(
queryset=Site.objects.all(),
to_field_name='slug',
widget=APISelectMultiple(
api_url="/api/dcim/sites/",
value_field="slug",
filter_for={
'power_panel_id': 'site',
2019-03-12 12:05:58 -04:00
'rack_id': 'site',
}
)
)
power_panel_id = FilterChoiceField(
queryset=PowerPanel.objects.all(),
label='Power panel',
null_label='-- None --',
widget=APISelectMultiple(
api_url="/api/dcim/power-panels/",
null_option=True,
)
)
2019-03-12 12:05:58 -04:00
rack_id = FilterChoiceField(
queryset=Rack.objects.all(),
label='Rack',
null_label='-- None --',
widget=APISelectMultiple(
api_url="/api/dcim/racks/",
null_option=True,
)
)
status = forms.MultipleChoiceField(
choices=POWERFEED_STATUS_CHOICES,
required=False,
widget=StaticSelect2Multiple()
)
type = forms.ChoiceField(
choices=add_blank_choice(POWERFEED_TYPE_CHOICES),
required=False,
widget=StaticSelect2()
)
supply = forms.ChoiceField(
choices=add_blank_choice(POWERFEED_SUPPLY_CHOICES),
required=False,
widget=StaticSelect2()
)
phase = forms.ChoiceField(
choices=add_blank_choice(POWERFEED_PHASE_CHOICES),
required=False,
widget=StaticSelect2()
)
2019-04-11 10:49:43 -04:00
voltage = forms.IntegerField(
required=False
)
amperage = forms.IntegerField(
required=False
)
2019-06-17 14:52:11 -04:00
max_utilization = forms.IntegerField(
2019-04-11 10:49:43 -04:00
required=False
)