mirror of
https://github.com/netbox-community/netbox.git
synced 2024-05-10 07:54:54 +00:00
4477 lines
123 KiB
Python
4477 lines
123 KiB
Python
import re
|
|
|
|
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
|
|
from django.core.exceptions import ObjectDoesNotExist
|
|
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
|
|
|
|
from circuits.models import Circuit, Provider
|
|
from extras.forms import (
|
|
AddRemoveTagsForm, CustomFieldForm, CustomFieldBulkEditForm, CustomFieldFilterForm, LocalConfigContextFilterForm
|
|
)
|
|
from ipam.constants import BGP_ASN_MAX, BGP_ASN_MIN
|
|
from ipam.models import IPAddress, VLAN
|
|
from tenancy.forms import TenancyFilterForm, TenancyForm
|
|
from tenancy.models import Tenant, TenantGroup
|
|
from utilities.forms import (
|
|
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,
|
|
)
|
|
from virtualization.models import Cluster, ClusterGroup, VirtualMachine
|
|
from .choices import *
|
|
from .constants import *
|
|
from .models import (
|
|
Cable, DeviceBay, DeviceBayTemplate, ConsolePort, ConsolePortTemplate, ConsoleServerPort, ConsoleServerPortTemplate,
|
|
Device, DeviceRole, DeviceType, FrontPort, FrontPortTemplate, Interface, InterfaceTemplate, Manufacturer,
|
|
InventoryItem, Platform, PowerFeed, PowerOutlet, PowerOutletTemplate, PowerPanel, PowerPort, PowerPortTemplate,
|
|
Rack, RackGroup, RackReservation, RackRole, RearPort, RearPortTemplate, Region, Site, VirtualChassis,
|
|
)
|
|
|
|
DEVICE_BY_PK_RE = r'{\d+\}'
|
|
|
|
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)
|
|
"""
|
|
|
|
|
|
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 DeviceComponentFilterForm(BootstrapMixin, forms.Form):
|
|
|
|
field_order = [
|
|
'q', 'region', 'site'
|
|
]
|
|
q = forms.CharField(
|
|
required=False,
|
|
label='Search'
|
|
)
|
|
region = FilterChoiceField(
|
|
queryset=Region.objects.all(),
|
|
to_field_name='slug',
|
|
required=False,
|
|
widget=APISelectMultiple(
|
|
api_url='/api/dcim/regions/',
|
|
value_field='slug',
|
|
filter_for={
|
|
'site': 'region'
|
|
}
|
|
)
|
|
)
|
|
site = FilterChoiceField(
|
|
queryset=Site.objects.all(),
|
|
to_field_name='slug',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/sites/",
|
|
value_field="slug"
|
|
)
|
|
)
|
|
|
|
|
|
class InterfaceCommonForm:
|
|
|
|
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'] == InterfaceModeChoices.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'] == InterfaceModeChoices.MODE_TAGGED_ALL:
|
|
self.cleaned_data['tagged_vlans'] = []
|
|
|
|
# Validate tagged VLANs; must be a global VLAN or in the same site
|
|
elif self.cleaned_data['mode'] == InterfaceModeChoices.MODE_TAGGED:
|
|
valid_sites = [None, self.cleaned_data['device'].site]
|
|
invalid_vlans = [str(v) for v in tagged_vlans if v.site not in valid_sites]
|
|
|
|
if invalid_vlans:
|
|
raise forms.ValidationError({
|
|
'tagged_vlans': "The tagged VLANs ({}) must belong to the same site as the interface's parent "
|
|
"device/VM, or they must be global".format(', '.join(invalid_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
|
|
|
|
|
|
#
|
|
# Regions
|
|
#
|
|
|
|
class RegionForm(BootstrapMixin, forms.ModelForm):
|
|
slug = SlugField()
|
|
|
|
class Meta:
|
|
model = Region
|
|
fields = [
|
|
'parent', 'name', 'slug',
|
|
]
|
|
widgets = {
|
|
'parent': APISelect(
|
|
api_url="/api/dcim/regions/"
|
|
)
|
|
}
|
|
|
|
|
|
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
|
|
help_texts = {
|
|
'name': 'Region name',
|
|
'slug': 'URL-friendly slug',
|
|
}
|
|
|
|
|
|
class RegionFilterForm(BootstrapMixin, forms.Form):
|
|
model = Site
|
|
q = forms.CharField(
|
|
required=False,
|
|
label='Search'
|
|
)
|
|
|
|
|
|
#
|
|
# Sites
|
|
#
|
|
|
|
class SiteForm(BootstrapMixin, TenancyForm, CustomFieldForm):
|
|
region = TreeNodeChoiceField(
|
|
queryset=Region.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/dcim/regions/"
|
|
)
|
|
)
|
|
slug = SlugField()
|
|
comments = CommentField()
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = Site
|
|
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',
|
|
]
|
|
widgets = {
|
|
'physical_address': SmallTextarea(
|
|
attrs={
|
|
'rows': 3,
|
|
}
|
|
),
|
|
'shipping_address': SmallTextarea(
|
|
attrs={
|
|
'rows': 3,
|
|
}
|
|
),
|
|
'status': StaticSelect2(),
|
|
'time_zone': StaticSelect2(),
|
|
}
|
|
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)",
|
|
'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)"
|
|
}
|
|
|
|
|
|
class SiteCSVForm(forms.ModelForm):
|
|
status = CSVChoiceField(
|
|
choices=SiteStatusChoices,
|
|
required=False,
|
|
help_text='Operational status'
|
|
)
|
|
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.',
|
|
}
|
|
)
|
|
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.',
|
|
}
|
|
)
|
|
|
|
class Meta:
|
|
model = Site
|
|
fields = Site.csv_headers
|
|
help_texts = {
|
|
'name': 'Site name',
|
|
'slug': 'URL-friendly slug',
|
|
'asn': '32-bit autonomous system number',
|
|
}
|
|
|
|
|
|
class SiteBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=Site.objects.all(),
|
|
widget=forms.MultipleHiddenInput
|
|
)
|
|
status = forms.ChoiceField(
|
|
choices=add_blank_choice(SiteStatusChoices),
|
|
required=False,
|
|
initial='',
|
|
widget=StaticSelect2()
|
|
)
|
|
region = TreeNodeChoiceField(
|
|
queryset=Region.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/dcim/regions/"
|
|
)
|
|
)
|
|
tenant = forms.ModelChoiceField(
|
|
queryset=Tenant.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/tenancy/tenants",
|
|
)
|
|
)
|
|
asn = forms.IntegerField(
|
|
min_value=BGP_ASN_MIN,
|
|
max_value=BGP_ASN_MAX,
|
|
required=False,
|
|
label='ASN'
|
|
)
|
|
description = forms.CharField(
|
|
max_length=100,
|
|
required=False
|
|
)
|
|
time_zone = TimeZoneFormField(
|
|
choices=add_blank_choice(TimeZoneFormField().choices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = [
|
|
'region', 'tenant', 'asn', 'description', 'time_zone',
|
|
]
|
|
|
|
|
|
class SiteFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldFilterForm):
|
|
model = Site
|
|
field_order = ['q', 'status', 'region', 'tenant_group', 'tenant']
|
|
q = forms.CharField(
|
|
required=False,
|
|
label='Search'
|
|
)
|
|
status = forms.MultipleChoiceField(
|
|
choices=SiteStatusChoices,
|
|
required=False,
|
|
widget=StaticSelect2Multiple()
|
|
)
|
|
region = forms.ModelMultipleChoiceField(
|
|
queryset=Region.objects.all(),
|
|
to_field_name='slug',
|
|
required=False,
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/regions/",
|
|
value_field="slug",
|
|
)
|
|
)
|
|
|
|
|
|
#
|
|
# Rack groups
|
|
#
|
|
|
|
class RackGroupForm(BootstrapMixin, forms.ModelForm):
|
|
slug = SlugField()
|
|
|
|
class Meta:
|
|
model = RackGroup
|
|
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):
|
|
region = FilterChoiceField(
|
|
queryset=Region.objects.all(),
|
|
to_field_name='slug',
|
|
required=False,
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/regions/",
|
|
value_field="slug",
|
|
filter_for={
|
|
'site': 'region'
|
|
}
|
|
)
|
|
)
|
|
site = FilterChoiceField(
|
|
queryset=Site.objects.all(),
|
|
to_field_name='slug',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/sites/",
|
|
value_field="slug",
|
|
)
|
|
)
|
|
|
|
|
|
#
|
|
# Rack roles
|
|
#
|
|
|
|
class RackRoleForm(BootstrapMixin, forms.ModelForm):
|
|
slug = SlugField()
|
|
|
|
class Meta:
|
|
model = RackRole
|
|
fields = [
|
|
'name', 'slug', 'color', 'description',
|
|
]
|
|
|
|
|
|
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)'
|
|
}
|
|
|
|
|
|
#
|
|
# 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/',
|
|
)
|
|
)
|
|
comments = CommentField()
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = Rack
|
|
fields = [
|
|
'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',
|
|
]
|
|
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',
|
|
}
|
|
),
|
|
'status': StaticSelect2(),
|
|
'role': APISelect(
|
|
api_url="/api/dcim/rack-roles/"
|
|
),
|
|
'type': StaticSelect2(),
|
|
'width': StaticSelect2(),
|
|
'outer_unit': StaticSelect2(),
|
|
}
|
|
|
|
|
|
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.',
|
|
}
|
|
)
|
|
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=RackStatusChoices,
|
|
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=RackTypeChoices,
|
|
required=False,
|
|
help_text='Rack type'
|
|
)
|
|
width = forms.ChoiceField(
|
|
choices=RackWidthChoices,
|
|
help_text='Rail-to-rail width (in inches)'
|
|
)
|
|
outer_unit = CSVChoiceField(
|
|
choices=RackDimensionUnitChoices,
|
|
required=False,
|
|
help_text='Unit for outer dimensions'
|
|
)
|
|
|
|
class Meta:
|
|
model = Rack
|
|
fields = Rack.csv_headers
|
|
help_texts = {
|
|
'name': 'Rack name',
|
|
'u_height': 'Height in rack units',
|
|
}
|
|
|
|
def clean(self):
|
|
|
|
super().clean()
|
|
|
|
site = self.cleaned_data.get('site')
|
|
group_name = self.cleaned_data.get('group_name')
|
|
name = self.cleaned_data.get('name')
|
|
facility_id = self.cleaned_data.get('facility_id')
|
|
|
|
# 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))
|
|
|
|
# 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)
|
|
)
|
|
|
|
|
|
class RackBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=Rack.objects.all(),
|
|
widget=forms.MultipleHiddenInput
|
|
)
|
|
site = forms.ModelChoiceField(
|
|
queryset=Site.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/dcim/sites",
|
|
filter_for={
|
|
'group': 'site_id',
|
|
}
|
|
)
|
|
)
|
|
group = forms.ModelChoiceField(
|
|
queryset=RackGroup.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/dcim/rack-groups",
|
|
)
|
|
)
|
|
tenant = forms.ModelChoiceField(
|
|
queryset=Tenant.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/tenancy/tenants",
|
|
)
|
|
)
|
|
status = forms.ChoiceField(
|
|
choices=add_blank_choice(RackStatusChoices),
|
|
required=False,
|
|
initial='',
|
|
widget=StaticSelect2()
|
|
)
|
|
role = forms.ModelChoiceField(
|
|
queryset=RackRole.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/dcim/rack-roles",
|
|
)
|
|
)
|
|
serial = forms.CharField(
|
|
max_length=50,
|
|
required=False,
|
|
label='Serial Number'
|
|
)
|
|
asset_tag = forms.CharField(
|
|
max_length=50,
|
|
required=False
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(RackTypeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
width = forms.ChoiceField(
|
|
choices=add_blank_choice(RackWidthChoices),
|
|
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(RackDimensionUnitChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
comments = CommentField(
|
|
widget=SmallTextarea
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = [
|
|
'group', 'tenant', 'role', 'serial', 'asset_tag', 'outer_width', 'outer_depth', 'outer_unit', 'comments',
|
|
]
|
|
|
|
|
|
class RackFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldFilterForm):
|
|
model = Rack
|
|
field_order = ['q', 'region', 'site', 'group_id', 'status', 'role', 'tenant_group', 'tenant']
|
|
q = forms.CharField(
|
|
required=False,
|
|
label='Search'
|
|
)
|
|
region = FilterChoiceField(
|
|
queryset=Region.objects.all(),
|
|
to_field_name='slug',
|
|
required=False,
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/regions/",
|
|
value_field="slug",
|
|
filter_for={
|
|
'site': 'region'
|
|
}
|
|
)
|
|
)
|
|
site = FilterChoiceField(
|
|
queryset=Site.objects.all(),
|
|
to_field_name='slug',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/sites/",
|
|
value_field="slug",
|
|
filter_for={
|
|
'group_id': 'site'
|
|
}
|
|
)
|
|
)
|
|
group_id = FilterChoiceField(
|
|
queryset=RackGroup.objects.prefetch_related(
|
|
'site'
|
|
),
|
|
label='Rack group',
|
|
null_label='-- None --',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/rack-groups/",
|
|
null_option=True
|
|
)
|
|
)
|
|
status = forms.MultipleChoiceField(
|
|
choices=RackStatusChoices,
|
|
required=False,
|
|
widget=StaticSelect2Multiple()
|
|
)
|
|
role = FilterChoiceField(
|
|
queryset=RackRole.objects.all(),
|
|
to_field_name='slug',
|
|
null_label='-- None --',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/rack-roles/",
|
|
value_field="slug",
|
|
null_option=True,
|
|
)
|
|
)
|
|
|
|
|
|
#
|
|
# Rack elevations
|
|
#
|
|
|
|
class RackElevationFilterForm(RackFilterForm):
|
|
field_order = ['q', 'region', 'site', 'group_id', 'id', 'status', 'role', 'tenant_group', 'tenant']
|
|
id = ChainedModelChoiceField(
|
|
queryset=Rack.objects.all(),
|
|
label='Rack',
|
|
chains=(
|
|
('site', 'site'),
|
|
('group_id', 'group_id'),
|
|
),
|
|
required=False,
|
|
widget=APISelectMultiple(
|
|
api_url='/api/dcim/racks/',
|
|
display_field='display_name',
|
|
)
|
|
)
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
|
|
# Filter the rack field based on the site and group
|
|
self.fields['site'].widget.add_filter_for('id', 'site')
|
|
self.fields['group_id'].widget.add_filter_for('id', 'group_id')
|
|
|
|
|
|
#
|
|
# Rack reservations
|
|
#
|
|
|
|
class RackReservationForm(BootstrapMixin, TenancyForm, forms.ModelForm):
|
|
units = SimpleArrayField(
|
|
base_field=forms.IntegerField(),
|
|
widget=ArrayFieldSelectMultiple(
|
|
attrs={
|
|
'size': 10,
|
|
}
|
|
)
|
|
)
|
|
user = forms.ModelChoiceField(
|
|
queryset=User.objects.order_by(
|
|
'username'
|
|
),
|
|
widget=StaticSelect2()
|
|
)
|
|
|
|
class Meta:
|
|
model = RackReservation
|
|
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):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=RackReservation.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
user = forms.ModelChoiceField(
|
|
queryset=User.objects.order_by(
|
|
'username'
|
|
),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
tenant = forms.ModelChoiceField(
|
|
queryset=Tenant.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/tenancy/tenant",
|
|
)
|
|
)
|
|
description = forms.CharField(
|
|
max_length=100,
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = []
|
|
|
|
|
|
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'),
|
|
label='Rack group',
|
|
null_label='-- None --',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/rack-groups/",
|
|
null_option=True,
|
|
)
|
|
)
|
|
|
|
|
|
#
|
|
# Manufacturers
|
|
#
|
|
|
|
class ManufacturerForm(BootstrapMixin, forms.ModelForm):
|
|
slug = SlugField()
|
|
|
|
class Meta:
|
|
model = Manufacturer
|
|
fields = [
|
|
'name', 'slug',
|
|
]
|
|
|
|
|
|
class ManufacturerCSVForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
model = Manufacturer
|
|
fields = Manufacturer.csv_headers
|
|
help_texts = {
|
|
'name': 'Manufacturer name',
|
|
'slug': 'URL-friendly slug',
|
|
}
|
|
|
|
|
|
#
|
|
# Device types
|
|
#
|
|
|
|
class DeviceTypeForm(BootstrapMixin, CustomFieldForm):
|
|
slug = SlugField(
|
|
slug_source='model'
|
|
)
|
|
comments = CommentField()
|
|
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',
|
|
]
|
|
|
|
|
|
class DeviceTypeBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=DeviceType.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
manufacturer = forms.ModelChoiceField(
|
|
queryset=Manufacturer.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/dcim/manufactureres"
|
|
)
|
|
)
|
|
u_height = forms.IntegerField(
|
|
min_value=1,
|
|
required=False
|
|
)
|
|
is_full_depth = forms.NullBooleanField(
|
|
required=False,
|
|
widget=BulkEditNullBooleanSelect(),
|
|
label='Is full depth'
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = []
|
|
|
|
|
|
class DeviceTypeFilterForm(BootstrapMixin, CustomFieldFilterForm):
|
|
model = DeviceType
|
|
q = forms.CharField(
|
|
required=False,
|
|
label='Search'
|
|
)
|
|
manufacturer = FilterChoiceField(
|
|
queryset=Manufacturer.objects.all(),
|
|
to_field_name='slug',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/manufacturers/",
|
|
value_field="slug",
|
|
)
|
|
)
|
|
subdevice_role = forms.MultipleChoiceField(
|
|
choices=add_blank_choice(SubdeviceRoleChoices),
|
|
required=False,
|
|
widget=StaticSelect2Multiple()
|
|
)
|
|
console_ports = forms.NullBooleanField(
|
|
required=False,
|
|
label='Has console ports',
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
console_server_ports = forms.NullBooleanField(
|
|
required=False,
|
|
label='Has console server ports',
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
power_ports = forms.NullBooleanField(
|
|
required=False,
|
|
label='Has power ports',
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
power_outlets = forms.NullBooleanField(
|
|
required=False,
|
|
label='Has power outlets',
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
interfaces = forms.NullBooleanField(
|
|
required=False,
|
|
label='Has interfaces',
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
pass_through_ports = forms.NullBooleanField(
|
|
required=False,
|
|
label='Has pass-through ports',
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
|
|
|
|
#
|
|
# Device component templates
|
|
#
|
|
|
|
class ConsolePortTemplateForm(BootstrapMixin, forms.ModelForm):
|
|
|
|
class Meta:
|
|
model = ConsolePortTemplate
|
|
fields = [
|
|
'device_type', 'name', 'type',
|
|
]
|
|
widgets = {
|
|
'device_type': forms.HiddenInput(),
|
|
}
|
|
|
|
|
|
class ConsolePortTemplateCreateForm(ComponentForm):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=ConsolePortTypeChoices,
|
|
widget=StaticSelect2()
|
|
)
|
|
|
|
|
|
class ConsoleServerPortTemplateForm(BootstrapMixin, forms.ModelForm):
|
|
|
|
class Meta:
|
|
model = ConsoleServerPortTemplate
|
|
fields = [
|
|
'device_type', 'name', 'type',
|
|
]
|
|
widgets = {
|
|
'device_type': forms.HiddenInput(),
|
|
}
|
|
|
|
|
|
class ConsoleServerPortTemplateCreateForm(ComponentForm):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(ConsolePortTypeChoices),
|
|
widget=StaticSelect2()
|
|
)
|
|
|
|
|
|
class PowerPortTemplateForm(BootstrapMixin, forms.ModelForm):
|
|
|
|
class Meta:
|
|
model = PowerPortTemplate
|
|
fields = [
|
|
'device_type', 'name', 'type', 'maximum_draw', 'allocated_draw',
|
|
]
|
|
widgets = {
|
|
'device_type': forms.HiddenInput(),
|
|
}
|
|
|
|
|
|
class PowerPortTemplateCreateForm(ComponentForm):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerPortTypeChoices),
|
|
required=False
|
|
)
|
|
maximum_draw = forms.IntegerField(
|
|
min_value=1,
|
|
required=False,
|
|
help_text="Maximum power draw (watts)"
|
|
)
|
|
allocated_draw = forms.IntegerField(
|
|
min_value=1,
|
|
required=False,
|
|
help_text="Allocated power draw (watts)"
|
|
)
|
|
|
|
|
|
class PowerOutletTemplateForm(BootstrapMixin, forms.ModelForm):
|
|
|
|
class Meta:
|
|
model = PowerOutletTemplate
|
|
fields = [
|
|
'device_type', 'name', 'type', 'power_port', 'feed_leg',
|
|
]
|
|
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'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerOutletTypeChoices),
|
|
required=False
|
|
)
|
|
power_port = forms.ModelChoiceField(
|
|
queryset=PowerPortTemplate.objects.all(),
|
|
required=False
|
|
)
|
|
feed_leg = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerOutletFeedLegChoices),
|
|
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
|
|
fields = [
|
|
'device_type', 'name', 'type', 'mgmt_only',
|
|
]
|
|
widgets = {
|
|
'device_type': forms.HiddenInput(),
|
|
'type': StaticSelect2(),
|
|
}
|
|
|
|
|
|
class InterfaceTemplateCreateForm(ComponentForm):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=InterfaceTypeChoices,
|
|
widget=StaticSelect2()
|
|
)
|
|
mgmt_only = forms.BooleanField(
|
|
required=False,
|
|
label='Management only'
|
|
)
|
|
|
|
|
|
class InterfaceTemplateBulkEditForm(BootstrapMixin, BulkEditForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=InterfaceTemplate.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(InterfaceTypeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
mgmt_only = forms.NullBooleanField(
|
|
required=False,
|
|
widget=BulkEditNullBooleanSelect,
|
|
label='Management only'
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = []
|
|
|
|
|
|
class FrontPortTemplateForm(BootstrapMixin, forms.ModelForm):
|
|
|
|
class Meta:
|
|
model = FrontPortTemplate
|
|
fields = [
|
|
'device_type', 'name', 'type', 'rear_port', 'rear_port_position',
|
|
]
|
|
widgets = {
|
|
'device_type': forms.HiddenInput(),
|
|
'rear_port': StaticSelect2(),
|
|
}
|
|
|
|
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
|
|
)
|
|
|
|
|
|
class FrontPortTemplateCreateForm(ComponentForm):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=PortTypeChoices,
|
|
widget=StaticSelect2()
|
|
)
|
|
rear_port_set = forms.MultipleChoiceField(
|
|
choices=[],
|
|
label='Rear ports',
|
|
help_text='Select one rear port assignment for each front port being created.',
|
|
)
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
# 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()
|
|
]
|
|
|
|
# Populate rear port choices
|
|
choices = []
|
|
rear_ports = RearPortTemplate.objects.filter(device_type=self.parent)
|
|
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):
|
|
|
|
class Meta:
|
|
model = RearPortTemplate
|
|
fields = [
|
|
'device_type', 'name', 'type', 'positions',
|
|
]
|
|
widgets = {
|
|
'device_type': forms.HiddenInput(),
|
|
'type': StaticSelect2(),
|
|
}
|
|
|
|
|
|
class RearPortTemplateCreateForm(ComponentForm):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=PortTypeChoices,
|
|
widget=StaticSelect2(),
|
|
)
|
|
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
|
|
fields = [
|
|
'device_type', 'name',
|
|
]
|
|
widgets = {
|
|
'device_type': forms.HiddenInput(),
|
|
}
|
|
|
|
|
|
class DeviceBayTemplateCreateForm(ComponentForm):
|
|
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):
|
|
|
|
class Meta:
|
|
model = ConsolePortTemplate
|
|
fields = [
|
|
'device_type', 'name', 'type',
|
|
]
|
|
|
|
|
|
class ConsoleServerPortTemplateImportForm(ComponentTemplateImportForm):
|
|
|
|
class Meta:
|
|
model = ConsoleServerPortTemplate
|
|
fields = [
|
|
'device_type', 'name', 'type',
|
|
]
|
|
|
|
|
|
class PowerPortTemplateImportForm(ComponentTemplateImportForm):
|
|
|
|
class Meta:
|
|
model = PowerPortTemplate
|
|
fields = [
|
|
'device_type', 'name', 'type', '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', 'type', 'power_port', 'feed_leg',
|
|
]
|
|
|
|
|
|
class InterfaceTemplateImportForm(ComponentTemplateImportForm):
|
|
type = forms.ChoiceField(
|
|
choices=InterfaceTypeChoices.CHOICES
|
|
)
|
|
|
|
class Meta:
|
|
model = InterfaceTemplate
|
|
fields = [
|
|
'device_type', 'name', 'type', 'mgmt_only',
|
|
]
|
|
|
|
|
|
class FrontPortTemplateImportForm(ComponentTemplateImportForm):
|
|
type = forms.ChoiceField(
|
|
choices=PortTypeChoices.CHOICES
|
|
)
|
|
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',
|
|
]
|
|
|
|
|
|
class RearPortTemplateImportForm(ComponentTemplateImportForm):
|
|
type = forms.ChoiceField(
|
|
choices=PortTypeChoices.CHOICES
|
|
)
|
|
|
|
class Meta:
|
|
model = RearPortTemplate
|
|
fields = [
|
|
'device_type', 'name', 'type', 'positions',
|
|
]
|
|
|
|
|
|
class DeviceBayTemplateImportForm(ComponentTemplateImportForm):
|
|
|
|
class Meta:
|
|
model = DeviceBayTemplate
|
|
fields = [
|
|
'device_type', 'name',
|
|
]
|
|
|
|
|
|
#
|
|
# Device roles
|
|
#
|
|
|
|
class DeviceRoleForm(BootstrapMixin, forms.ModelForm):
|
|
slug = SlugField()
|
|
|
|
class Meta:
|
|
model = DeviceRole
|
|
fields = [
|
|
'name', 'slug', 'color', 'vm_role', 'description',
|
|
]
|
|
|
|
|
|
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)'
|
|
}
|
|
|
|
|
|
#
|
|
# Platforms
|
|
#
|
|
|
|
class PlatformForm(BootstrapMixin, forms.ModelForm):
|
|
slug = SlugField(
|
|
max_length=64
|
|
)
|
|
|
|
class Meta:
|
|
model = Platform
|
|
fields = [
|
|
'name', 'slug', 'manufacturer', 'napalm_driver', 'napalm_args',
|
|
]
|
|
widgets = {
|
|
'manufacturer': APISelect(
|
|
api_url="/api/dcim/manufacturers/"
|
|
),
|
|
'napalm_args': SmallTextarea(),
|
|
}
|
|
|
|
|
|
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',
|
|
}
|
|
|
|
|
|
#
|
|
# 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'
|
|
}
|
|
)
|
|
)
|
|
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}}/elevation/',
|
|
disabled_indicator='device'
|
|
)
|
|
)
|
|
manufacturer = forms.ModelChoiceField(
|
|
queryset=Manufacturer.objects.all(),
|
|
widget=APISelect(
|
|
api_url="/api/dcim/manufacturers/",
|
|
filter_for={
|
|
'device_type': 'manufacturer_id',
|
|
'platform': 'manufacturer_id'
|
|
}
|
|
)
|
|
)
|
|
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/',
|
|
)
|
|
)
|
|
comments = CommentField()
|
|
tags = TagField(required=False)
|
|
local_context_data = JSONField(
|
|
required=False,
|
|
label=''
|
|
)
|
|
|
|
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'
|
|
]
|
|
help_texts = {
|
|
'device_role': "The function this device serves",
|
|
'serial': "Chassis serial number",
|
|
'local_context_data': "Local config context data overwrites all source contexts in the final rendered "
|
|
"config context",
|
|
}
|
|
widgets = {
|
|
'face': StaticSelect2(
|
|
filter_for={
|
|
'position': 'face'
|
|
}
|
|
),
|
|
'device_role': APISelect(
|
|
api_url='/api/dcim/device-roles/'
|
|
),
|
|
'status': StaticSelect2(),
|
|
'platform': APISelect(
|
|
api_url="/api/dcim/platforms/",
|
|
additional_query_params={
|
|
"manufacturer_id": "null"
|
|
}
|
|
),
|
|
'primary_ip4': StaticSelect2(),
|
|
'primary_ip6': StaticSelect2(),
|
|
}
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
# 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
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
if self.instance.pk:
|
|
|
|
# Compile list of choices for primary IPv4 and IPv6 addresses
|
|
for family in [4, 6]:
|
|
ip_choices = [(None, '---------')]
|
|
|
|
# Gather PKs of all interfaces belonging to this 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
|
|
|
|
# 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.
|
|
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)
|
|
)
|
|
|
|
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
|
|
|
|
# Rack position
|
|
pk = self.instance.pk if self.instance.pk else None
|
|
try:
|
|
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)
|
|
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)
|
|
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
|
|
|
|
|
|
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=DeviceStatusChoices,
|
|
help_text='Operational status'
|
|
)
|
|
|
|
class Meta:
|
|
fields = []
|
|
model = Device
|
|
help_texts = {
|
|
'name': 'Device name',
|
|
}
|
|
|
|
def clean(self):
|
|
|
|
super().clean()
|
|
|
|
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:
|
|
raise forms.ValidationError("Device type {} {} not found".format(manufacturer, model_name))
|
|
|
|
|
|
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.',
|
|
}
|
|
)
|
|
rack_group = forms.CharField(
|
|
required=False,
|
|
help_text='Parent rack\'s group (if any)'
|
|
)
|
|
rack_name = forms.CharField(
|
|
required=False,
|
|
help_text='Name of parent rack'
|
|
)
|
|
face = CSVChoiceField(
|
|
choices=DeviceFaceChoices,
|
|
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')
|
|
rack_group = self.cleaned_data.get('rack_group')
|
|
rack_name = self.cleaned_data.get('rack_name')
|
|
|
|
# Validate rack
|
|
if site and rack_group and rack_name:
|
|
try:
|
|
self.instance.rack = Rack.objects.get(site=site, group__name=rack_group, name=rack_name)
|
|
except Rack.DoesNotExist:
|
|
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:
|
|
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))
|
|
|
|
|
|
class DeviceBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=Device.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
device_type = forms.ModelChoiceField(
|
|
queryset=DeviceType.objects.all(),
|
|
required=False,
|
|
label='Type',
|
|
widget=APISelect(
|
|
api_url="/api/dcim/device-types/",
|
|
display_field='display_name'
|
|
)
|
|
)
|
|
device_role = forms.ModelChoiceField(
|
|
queryset=DeviceRole.objects.all(),
|
|
required=False,
|
|
label='Role',
|
|
widget=APISelect(
|
|
api_url="/api/dcim/device-roles/"
|
|
)
|
|
)
|
|
tenant = forms.ModelChoiceField(
|
|
queryset=Tenant.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/tenancy/tenants/"
|
|
)
|
|
)
|
|
platform = forms.ModelChoiceField(
|
|
queryset=Platform.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/dcim/platforms/"
|
|
)
|
|
)
|
|
status = forms.ChoiceField(
|
|
choices=add_blank_choice(DeviceStatusChoices),
|
|
required=False,
|
|
initial='',
|
|
widget=StaticSelect2()
|
|
)
|
|
serial = forms.CharField(
|
|
max_length=50,
|
|
required=False,
|
|
label='Serial Number'
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = [
|
|
'tenant', 'platform', 'serial',
|
|
]
|
|
|
|
|
|
class DeviceFilterForm(BootstrapMixin, LocalConfigContextFilterForm, TenancyFilterForm, CustomFieldFilterForm):
|
|
model = Device
|
|
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',
|
|
]
|
|
q = forms.CharField(
|
|
required=False,
|
|
label='Search'
|
|
)
|
|
region = FilterChoiceField(
|
|
queryset=Region.objects.all(),
|
|
to_field_name='slug',
|
|
required=False,
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/regions/",
|
|
value_field="slug",
|
|
filter_for={
|
|
'site': 'region'
|
|
}
|
|
)
|
|
)
|
|
site = FilterChoiceField(
|
|
queryset=Site.objects.all(),
|
|
to_field_name='slug',
|
|
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(
|
|
'site'
|
|
),
|
|
label='Rack group',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/rack-groups/",
|
|
filter_for={
|
|
'rack_id': 'group_id',
|
|
}
|
|
)
|
|
)
|
|
rack_id = FilterChoiceField(
|
|
queryset=Rack.objects.all(),
|
|
label='Rack',
|
|
null_label='-- None --',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/racks/",
|
|
null_option=True,
|
|
)
|
|
)
|
|
role = FilterChoiceField(
|
|
queryset=DeviceRole.objects.all(),
|
|
to_field_name='slug',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/device-roles/",
|
|
value_field="slug",
|
|
)
|
|
)
|
|
manufacturer_id = FilterChoiceField(
|
|
queryset=Manufacturer.objects.all(),
|
|
label='Manufacturer',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/manufacturers/",
|
|
filter_for={
|
|
'device_type_id': 'manufacturer_id',
|
|
}
|
|
)
|
|
)
|
|
device_type_id = FilterChoiceField(
|
|
queryset=DeviceType.objects.prefetch_related(
|
|
'manufacturer'
|
|
),
|
|
label='Model',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/device-types/",
|
|
display_field="model",
|
|
)
|
|
)
|
|
platform = FilterChoiceField(
|
|
queryset=Platform.objects.all(),
|
|
to_field_name='slug',
|
|
null_label='-- None --',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/platforms/",
|
|
value_field="slug",
|
|
null_option=True,
|
|
)
|
|
)
|
|
status = forms.MultipleChoiceField(
|
|
choices=DeviceStatusChoices,
|
|
required=False,
|
|
widget=StaticSelect2Multiple()
|
|
)
|
|
mac_address = forms.CharField(
|
|
required=False,
|
|
label='MAC address'
|
|
)
|
|
has_primary_ip = forms.NullBooleanField(
|
|
required=False,
|
|
label='Has a primary IP',
|
|
widget=StaticSelect2(
|
|
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',
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
console_server_ports = forms.NullBooleanField(
|
|
required=False,
|
|
label='Has console server ports',
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
power_ports = forms.NullBooleanField(
|
|
required=False,
|
|
label='Has power ports',
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
power_outlets = forms.NullBooleanField(
|
|
required=False,
|
|
label='Has power outlets',
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
interfaces = forms.NullBooleanField(
|
|
required=False,
|
|
label='Has interfaces',
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
pass_through_ports = forms.NullBooleanField(
|
|
required=False,
|
|
label='Has pass-through ports',
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
|
|
|
|
#
|
|
# Bulk device component creation
|
|
#
|
|
|
|
class DeviceBulkAddComponentForm(BootstrapMixin, forms.Form):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=Device.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
|
|
|
|
class DeviceBulkAddInterfaceForm(DeviceBulkAddComponentForm):
|
|
type = forms.ChoiceField(
|
|
choices=InterfaceTypeChoices,
|
|
widget=StaticSelect2()
|
|
)
|
|
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
|
|
)
|
|
|
|
|
|
#
|
|
# Console ports
|
|
#
|
|
|
|
|
|
class ConsolePortFilterForm(DeviceComponentFilterForm):
|
|
model = ConsolePort
|
|
|
|
|
|
class ConsolePortForm(BootstrapMixin, forms.ModelForm):
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = ConsolePort
|
|
fields = [
|
|
'device', 'name', 'type', 'description', 'tags',
|
|
]
|
|
widgets = {
|
|
'device': forms.HiddenInput(),
|
|
}
|
|
|
|
|
|
class ConsolePortCreateForm(ComponentForm):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(ConsolePortTypeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
description = forms.CharField(
|
|
max_length=100,
|
|
required=False
|
|
)
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
|
|
class ConsolePortCSVForm(forms.ModelForm):
|
|
device = FlexibleModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
to_field_name='name',
|
|
help_text='Name or ID of device',
|
|
error_messages={
|
|
'invalid_choice': 'Device not found.',
|
|
}
|
|
)
|
|
|
|
class Meta:
|
|
model = ConsolePort
|
|
fields = ConsolePort.csv_headers
|
|
|
|
|
|
#
|
|
# Console server ports
|
|
#
|
|
|
|
|
|
class ConsoleServerPortFilterForm(DeviceComponentFilterForm):
|
|
model = ConsoleServerPort
|
|
|
|
|
|
class ConsoleServerPortForm(BootstrapMixin, forms.ModelForm):
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = ConsoleServerPort
|
|
fields = [
|
|
'device', 'name', 'type', 'description', 'tags',
|
|
]
|
|
widgets = {
|
|
'device': forms.HiddenInput(),
|
|
}
|
|
|
|
|
|
class ConsoleServerPortCreateForm(ComponentForm):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(ConsolePortTypeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
description = forms.CharField(
|
|
max_length=100,
|
|
required=False
|
|
)
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
|
|
class ConsoleServerPortBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=ConsoleServerPort.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(ConsolePortTypeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
description = forms.CharField(
|
|
max_length=100,
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = [
|
|
'description',
|
|
]
|
|
|
|
|
|
class ConsoleServerPortBulkRenameForm(BulkRenameForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=ConsoleServerPort.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
|
|
|
|
class ConsoleServerPortBulkDisconnectForm(ConfirmationForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=ConsoleServerPort.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
|
|
|
|
class ConsoleServerPortCSVForm(forms.ModelForm):
|
|
device = FlexibleModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
to_field_name='name',
|
|
help_text='Name or ID of device',
|
|
error_messages={
|
|
'invalid_choice': 'Device not found.',
|
|
}
|
|
)
|
|
|
|
class Meta:
|
|
model = ConsoleServerPort
|
|
fields = ConsoleServerPort.csv_headers
|
|
|
|
|
|
#
|
|
# Power ports
|
|
#
|
|
|
|
|
|
class PowerPortFilterForm(DeviceComponentFilterForm):
|
|
model = PowerPort
|
|
|
|
|
|
class PowerPortForm(BootstrapMixin, forms.ModelForm):
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = PowerPort
|
|
fields = [
|
|
'device', 'name', 'type', 'maximum_draw', 'allocated_draw', 'description', 'tags',
|
|
]
|
|
widgets = {
|
|
'device': forms.HiddenInput(),
|
|
}
|
|
|
|
|
|
class PowerPortCreateForm(ComponentForm):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerPortTypeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
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
|
|
)
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
|
|
class PowerPortCSVForm(forms.ModelForm):
|
|
device = FlexibleModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
to_field_name='name',
|
|
help_text='Name or ID of device',
|
|
error_messages={
|
|
'invalid_choice': 'Device not found.',
|
|
}
|
|
)
|
|
|
|
class Meta:
|
|
model = PowerPort
|
|
fields = PowerPort.csv_headers
|
|
|
|
|
|
#
|
|
# Power outlets
|
|
#
|
|
|
|
|
|
class PowerOutletFilterForm(DeviceComponentFilterForm):
|
|
model = PowerOutlet
|
|
|
|
|
|
class PowerOutletForm(BootstrapMixin, forms.ModelForm):
|
|
power_port = forms.ModelChoiceField(
|
|
queryset=PowerPort.objects.all(),
|
|
required=False
|
|
)
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = PowerOutlet
|
|
fields = [
|
|
'device', 'name', 'type', 'power_port', 'feed_leg', 'description', 'tags',
|
|
]
|
|
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
|
|
)
|
|
|
|
|
|
class PowerOutletCreateForm(ComponentForm):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerOutletTypeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
power_port = forms.ModelChoiceField(
|
|
queryset=PowerPort.objects.all(),
|
|
required=False
|
|
)
|
|
feed_leg = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerOutletFeedLegChoices),
|
|
required=False
|
|
)
|
|
description = forms.CharField(
|
|
max_length=100,
|
|
required=False
|
|
)
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
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)
|
|
|
|
|
|
class PowerOutletCSVForm(forms.ModelForm):
|
|
device = FlexibleModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
to_field_name='name',
|
|
help_text='Name or ID of device',
|
|
error_messages={
|
|
'invalid_choice': 'Device not found.',
|
|
}
|
|
)
|
|
power_port = FlexibleModelChoiceField(
|
|
queryset=PowerPort.objects.all(),
|
|
required=False,
|
|
to_field_name='name',
|
|
help_text='Name or ID of Power Port',
|
|
error_messages={
|
|
'invalid_choice': 'Power Port not found.',
|
|
}
|
|
)
|
|
feed_leg = CSVChoiceField(
|
|
choices=PowerOutletFeedLegChoices,
|
|
required=False,
|
|
)
|
|
|
|
class Meta:
|
|
model = PowerOutlet
|
|
fields = PowerOutlet.csv_headers
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
|
|
# Limit PowerPort choices to those belonging to this device (or VC master)
|
|
if self.is_bound:
|
|
try:
|
|
device = self.fields['device'].to_python(self.data['device'])
|
|
except forms.ValidationError:
|
|
device = None
|
|
else:
|
|
try:
|
|
device = self.instance.device
|
|
except Device.DoesNotExist:
|
|
device = None
|
|
|
|
if device:
|
|
self.fields['power_port'].queryset = PowerPort.objects.filter(
|
|
device__in=[device, device.get_vc_master()]
|
|
)
|
|
else:
|
|
self.fields['power_port'].queryset = PowerPort.objects.none()
|
|
|
|
|
|
class PowerOutletBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=PowerOutlet.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=PowerOutletTypeChoices,
|
|
required=False
|
|
)
|
|
feed_leg = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerOutletFeedLegChoices),
|
|
required=False,
|
|
)
|
|
power_port = forms.ModelChoiceField(
|
|
queryset=PowerPort.objects.all(),
|
|
required=False
|
|
)
|
|
description = forms.CharField(
|
|
max_length=100,
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = [
|
|
'type', '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):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=PowerOutlet.objects.all(),
|
|
widget=forms.MultipleHiddenInput
|
|
)
|
|
|
|
|
|
class PowerOutletBulkDisconnectForm(ConfirmationForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=PowerOutlet.objects.all(),
|
|
widget=forms.MultipleHiddenInput
|
|
)
|
|
|
|
|
|
#
|
|
# Interfaces
|
|
#
|
|
|
|
|
|
class InterfaceFilterForm(DeviceComponentFilterForm):
|
|
model = Interface
|
|
|
|
|
|
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
|
|
)
|
|
)
|
|
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = Interface
|
|
fields = [
|
|
'device', 'name', 'type', 'enabled', 'lag', 'mac_address', 'mtu', 'mgmt_only', 'description',
|
|
'mode', 'untagged_vlan', 'tagged_vlans', 'tags',
|
|
]
|
|
widgets = {
|
|
'device': forms.HiddenInput(),
|
|
'type': StaticSelect2(),
|
|
'lag': StaticSelect2(),
|
|
'mode': StaticSelect2(),
|
|
}
|
|
labels = {
|
|
'mode': '802.1Q Mode',
|
|
}
|
|
help_texts = {
|
|
'mode': INTERFACE_MODE_HELP_TEXT,
|
|
}
|
|
|
|
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=InterfaceTypeChoices.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=InterfaceTypeChoices.TYPE_LAG
|
|
)
|
|
|
|
|
|
class InterfaceCreateForm(InterfaceCommonForm, ComponentForm, forms.Form):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=InterfaceTypeChoices,
|
|
widget=StaticSelect2(),
|
|
)
|
|
enabled = forms.BooleanField(
|
|
required=False
|
|
)
|
|
lag = forms.ModelChoiceField(
|
|
queryset=Interface.objects.all(),
|
|
required=False,
|
|
label='Parent LAG',
|
|
widget=StaticSelect2(),
|
|
)
|
|
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,
|
|
label='Management only',
|
|
help_text='This interface is used only for out-of-band management'
|
|
)
|
|
description = forms.CharField(
|
|
max_length=100,
|
|
required=False
|
|
)
|
|
mode = forms.ChoiceField(
|
|
choices=add_blank_choice(InterfaceModeChoices),
|
|
required=False,
|
|
widget=StaticSelect2(),
|
|
)
|
|
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
|
|
)
|
|
)
|
|
|
|
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=InterfaceTypeChoices.TYPE_LAG
|
|
)
|
|
else:
|
|
self.fields['lag'].queryset = Interface.objects.none()
|
|
|
|
|
|
class InterfaceCSVForm(forms.ModelForm):
|
|
device = FlexibleModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
required=False,
|
|
to_field_name='name',
|
|
help_text='Name or ID of device',
|
|
error_messages={
|
|
'invalid_choice': 'Device not found.',
|
|
}
|
|
)
|
|
virtual_machine = FlexibleModelChoiceField(
|
|
queryset=VirtualMachine.objects.all(),
|
|
required=False,
|
|
to_field_name='name',
|
|
help_text='Name or ID of virtual machine',
|
|
error_messages={
|
|
'invalid_choice': 'Virtual machine not found.',
|
|
}
|
|
)
|
|
lag = FlexibleModelChoiceField(
|
|
queryset=Interface.objects.all(),
|
|
required=False,
|
|
to_field_name='name',
|
|
help_text='Name or ID of LAG interface',
|
|
error_messages={
|
|
'invalid_choice': 'LAG interface not found.',
|
|
}
|
|
)
|
|
type = CSVChoiceField(
|
|
choices=InterfaceTypeChoices,
|
|
)
|
|
mode = CSVChoiceField(
|
|
choices=InterfaceModeChoices,
|
|
required=False,
|
|
)
|
|
|
|
class Meta:
|
|
model = Interface
|
|
fields = Interface.csv_headers
|
|
|
|
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:
|
|
try:
|
|
device = self.fields['device'].to_python(self.data['device'])
|
|
except forms.ValidationError:
|
|
device = None
|
|
else:
|
|
device = self.instance.device
|
|
|
|
if device:
|
|
self.fields['lag'].queryset = Interface.objects.filter(
|
|
device__in=[device, device.get_vc_master()], type=InterfaceTypeChoices.TYPE_LAG
|
|
)
|
|
else:
|
|
self.fields['lag'].queryset = Interface.objects.none()
|
|
|
|
def clean_enabled(self):
|
|
# Make sure enabled is True when it's not included in the uploaded data
|
|
if 'enabled' not in self.data:
|
|
return True
|
|
else:
|
|
return self.cleaned_data['enabled']
|
|
|
|
|
|
class InterfaceBulkEditForm(InterfaceCommonForm, BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=Interface.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(InterfaceTypeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
enabled = forms.NullBooleanField(
|
|
required=False,
|
|
widget=BulkEditNullBooleanSelect()
|
|
)
|
|
lag = forms.ModelChoiceField(
|
|
queryset=Interface.objects.all(),
|
|
required=False,
|
|
label='Parent LAG',
|
|
widget=StaticSelect2()
|
|
)
|
|
mac_address = forms.CharField(
|
|
required=False,
|
|
label='MAC Address'
|
|
)
|
|
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(InterfaceModeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
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
|
|
)
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = [
|
|
'lag', 'mac_address', 'mtu', 'description', 'mode', 'untagged_vlan', 'tagged_vlans'
|
|
]
|
|
|
|
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=InterfaceTypeChoices.TYPE_LAG
|
|
)
|
|
else:
|
|
self.fields['lag'].choices = []
|
|
|
|
|
|
class InterfaceBulkRenameForm(BulkRenameForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=Interface.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
|
|
|
|
class InterfaceBulkDisconnectForm(ConfirmationForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=Interface.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
|
|
|
|
#
|
|
# Front pass-through ports
|
|
#
|
|
|
|
class FrontPortFilterForm(DeviceComponentFilterForm):
|
|
model = FrontPort
|
|
|
|
|
|
class FrontPortForm(BootstrapMixin, forms.ModelForm):
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = FrontPort
|
|
fields = [
|
|
'device', 'name', 'type', 'rear_port', 'rear_port_position', 'description', 'tags',
|
|
]
|
|
widgets = {
|
|
'device': forms.HiddenInput(),
|
|
'type': StaticSelect2(),
|
|
'rear_port': StaticSelect2(),
|
|
}
|
|
|
|
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
|
|
)
|
|
|
|
|
|
# TODO: Merge with FrontPortTemplateCreateForm to remove duplicate logic
|
|
class FrontPortCreateForm(ComponentForm):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=PortTypeChoices,
|
|
widget=StaticSelect2(),
|
|
)
|
|
rear_port_set = forms.MultipleChoiceField(
|
|
choices=[],
|
|
label='Rear ports',
|
|
help_text='Select one rear port assignment for each front port being created.',
|
|
)
|
|
description = forms.CharField(
|
|
required=False
|
|
)
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
|
|
# 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.frontports.all()
|
|
]
|
|
|
|
# Populate rear port choices
|
|
choices = []
|
|
rear_ports = RearPort.objects.filter(device=self.parent)
|
|
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 FrontPortCSVForm(forms.ModelForm):
|
|
device = FlexibleModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
to_field_name='name',
|
|
help_text='Name or ID of device',
|
|
error_messages={
|
|
'invalid_choice': 'Device not found.',
|
|
}
|
|
)
|
|
rear_port = FlexibleModelChoiceField(
|
|
queryset=RearPort.objects.all(),
|
|
to_field_name='name',
|
|
help_text='Name or ID of Rear Port',
|
|
error_messages={
|
|
'invalid_choice': 'Rear Port not found.',
|
|
}
|
|
)
|
|
type = CSVChoiceField(
|
|
choices=PortTypeChoices,
|
|
)
|
|
|
|
class Meta:
|
|
model = FrontPort
|
|
fields = FrontPort.csv_headers
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
|
|
# Limit RearPort choices to those belonging to this device (or VC master)
|
|
if self.is_bound:
|
|
try:
|
|
device = self.fields['device'].to_python(self.data['device'])
|
|
except forms.ValidationError:
|
|
device = None
|
|
else:
|
|
try:
|
|
device = self.instance.device
|
|
except Device.DoesNotExist:
|
|
device = None
|
|
|
|
if device:
|
|
self.fields['rear_port'].queryset = RearPort.objects.filter(
|
|
device__in=[device, device.get_vc_master()]
|
|
)
|
|
else:
|
|
self.fields['rear_port'].queryset = RearPort.objects.none()
|
|
|
|
|
|
class FrontPortBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=FrontPort.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(PortTypeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
description = forms.CharField(
|
|
max_length=100,
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = [
|
|
'description',
|
|
]
|
|
|
|
|
|
class FrontPortBulkRenameForm(BulkRenameForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=FrontPort.objects.all(),
|
|
widget=forms.MultipleHiddenInput
|
|
)
|
|
|
|
|
|
class FrontPortBulkDisconnectForm(ConfirmationForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=FrontPort.objects.all(),
|
|
widget=forms.MultipleHiddenInput
|
|
)
|
|
|
|
|
|
#
|
|
# Rear pass-through ports
|
|
#
|
|
|
|
class RearPortFilterForm(DeviceComponentFilterForm):
|
|
model = RearPort
|
|
|
|
|
|
class RearPortForm(BootstrapMixin, forms.ModelForm):
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = RearPort
|
|
fields = [
|
|
'device', 'name', 'type', 'positions', 'description', 'tags',
|
|
]
|
|
widgets = {
|
|
'device': forms.HiddenInput(),
|
|
'type': StaticSelect2(),
|
|
}
|
|
|
|
|
|
class RearPortCreateForm(ComponentForm):
|
|
name_pattern = ExpandableNameField(
|
|
label='Name'
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=PortTypeChoices,
|
|
widget=StaticSelect2(),
|
|
)
|
|
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
|
|
)
|
|
|
|
|
|
class RearPortCSVForm(forms.ModelForm):
|
|
device = FlexibleModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
to_field_name='name',
|
|
help_text='Name or ID of device',
|
|
error_messages={
|
|
'invalid_choice': 'Device not found.',
|
|
}
|
|
)
|
|
type = CSVChoiceField(
|
|
choices=PortTypeChoices,
|
|
)
|
|
|
|
class Meta:
|
|
model = RearPort
|
|
fields = RearPort.csv_headers
|
|
|
|
|
|
class RearPortBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=RearPort.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(PortTypeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
description = forms.CharField(
|
|
max_length=100,
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = [
|
|
'description',
|
|
]
|
|
|
|
|
|
class RearPortBulkRenameForm(BulkRenameForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=RearPort.objects.all(),
|
|
widget=forms.MultipleHiddenInput
|
|
)
|
|
|
|
|
|
class RearPortBulkDisconnectForm(ConfirmationForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=RearPort.objects.all(),
|
|
widget=forms.MultipleHiddenInput
|
|
)
|
|
|
|
|
|
#
|
|
# Cables
|
|
#
|
|
|
|
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,
|
|
widget=APISelect(
|
|
api_url='/api/dcim/sites/',
|
|
filter_for={
|
|
'termination_b_rack': 'site_id',
|
|
'termination_b_device': 'site_id',
|
|
}
|
|
)
|
|
)
|
|
termination_b_rack = ChainedModelChoiceField(
|
|
queryset=Rack.objects.all(),
|
|
chains=(
|
|
('site', 'termination_b_site'),
|
|
),
|
|
label='Rack',
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url='/api/dcim/racks/',
|
|
filter_for={
|
|
'termination_b_device': 'rack_id',
|
|
},
|
|
attrs={
|
|
'nullable': 'true',
|
|
}
|
|
)
|
|
)
|
|
termination_b_device = ChainedModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
chains=(
|
|
('site', 'termination_b_site'),
|
|
('rack', 'termination_b_rack'),
|
|
),
|
|
label='Device',
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url='/api/dcim/devices/',
|
|
display_field='display_name',
|
|
filter_for={
|
|
'termination_b_id': 'device_id',
|
|
}
|
|
)
|
|
)
|
|
|
|
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',
|
|
}
|
|
)
|
|
)
|
|
|
|
|
|
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',
|
|
)
|
|
)
|
|
|
|
|
|
class ConnectCableToCircuitTerminationForm(BootstrapMixin, ChainedFieldsMixin, forms.ModelForm):
|
|
termination_b_provider = forms.ModelChoiceField(
|
|
queryset=Provider.objects.all(),
|
|
label='Provider',
|
|
required=False,
|
|
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',
|
|
}
|
|
)
|
|
)
|
|
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',
|
|
widget=APISelect(
|
|
api_url='/api/circuits/circuit-terminations/',
|
|
disabled_indicator='cable',
|
|
display_field='term_side'
|
|
)
|
|
)
|
|
|
|
class Meta:
|
|
model = Cable
|
|
fields = [
|
|
'termination_b_provider', 'termination_b_site', 'termination_b_circuit', 'termination_b_id', 'type',
|
|
'status', 'label', 'color', 'length', 'length_unit',
|
|
]
|
|
|
|
|
|
class ConnectCableToPowerFeedForm(BootstrapMixin, ChainedFieldsMixin, forms.ModelForm):
|
|
termination_b_site = forms.ModelChoiceField(
|
|
queryset=Site.objects.all(),
|
|
label='Site',
|
|
required=False,
|
|
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',
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url='/api/dcim/power-panels/',
|
|
filter_for={
|
|
'termination_b_id': 'power_panel_id',
|
|
}
|
|
)
|
|
)
|
|
termination_b_id = forms.IntegerField(
|
|
label='Name',
|
|
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',
|
|
]
|
|
|
|
|
|
class CableForm(BootstrapMixin, forms.ModelForm):
|
|
|
|
class Meta:
|
|
model = Cable
|
|
fields = [
|
|
'type', 'status', 'label', 'color', 'length', 'length_unit',
|
|
]
|
|
|
|
|
|
class CableCSVForm(forms.ModelForm):
|
|
|
|
# Termination A
|
|
side_a_device = FlexibleModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
to_field_name='name',
|
|
help_text='Side A device name or ID',
|
|
error_messages={
|
|
'invalid_choice': 'Side A device not found',
|
|
}
|
|
)
|
|
side_a_type = forms.ModelChoiceField(
|
|
queryset=ContentType.objects.all(),
|
|
limit_choices_to=CABLE_TERMINATION_MODELS,
|
|
to_field_name='model',
|
|
help_text='Side A type'
|
|
)
|
|
side_a_name = forms.CharField(
|
|
help_text='Side A component'
|
|
)
|
|
|
|
# Termination B
|
|
side_b_device = FlexibleModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
to_field_name='name',
|
|
help_text='Side B device name or ID',
|
|
error_messages={
|
|
'invalid_choice': 'Side B device not found',
|
|
}
|
|
)
|
|
side_b_type = forms.ModelChoiceField(
|
|
queryset=ContentType.objects.all(),
|
|
limit_choices_to=CABLE_TERMINATION_MODELS,
|
|
to_field_name='model',
|
|
help_text='Side B type'
|
|
)
|
|
side_b_name = forms.CharField(
|
|
help_text='Side B component'
|
|
)
|
|
|
|
# Cable attributes
|
|
status = CSVChoiceField(
|
|
choices=CableStatusChoices,
|
|
required=False,
|
|
help_text='Connection status'
|
|
)
|
|
type = CSVChoiceField(
|
|
choices=CableTypeChoices,
|
|
required=False,
|
|
help_text='Cable type'
|
|
)
|
|
length_unit = CSVChoiceField(
|
|
choices=CableLengthUnitChoices,
|
|
required=False,
|
|
help_text='Length unit'
|
|
)
|
|
|
|
class Meta:
|
|
model = Cable
|
|
fields = [
|
|
'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',
|
|
]
|
|
help_texts = {
|
|
'color': 'RGB color in hexadecimal (e.g. 00ff00)'
|
|
}
|
|
|
|
# TODO: Merge the clean() methods for either end
|
|
def clean_side_a_name(self):
|
|
|
|
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
|
|
|
|
model = content_type.model_class()
|
|
try:
|
|
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)
|
|
)
|
|
|
|
self.instance.termination_a = termination_object
|
|
return termination_object
|
|
|
|
def clean_side_b_name(self):
|
|
|
|
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
|
|
|
|
model = content_type.model_class()
|
|
try:
|
|
termination_object = model.objects.get(
|
|
device=device,
|
|
name=name
|
|
)
|
|
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 ''
|
|
|
|
|
|
class CableBulkEditForm(BootstrapMixin, BulkEditForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=Cable.objects.all(),
|
|
widget=forms.MultipleHiddenInput
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(CableTypeChoices),
|
|
required=False,
|
|
initial='',
|
|
widget=StaticSelect2()
|
|
)
|
|
status = forms.ChoiceField(
|
|
choices=add_blank_choice(CableStatusChoices),
|
|
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(CableLengthUnitChoices),
|
|
required=False,
|
|
initial='',
|
|
widget=StaticSelect2()
|
|
)
|
|
|
|
class Meta:
|
|
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"
|
|
})
|
|
|
|
|
|
class CableFilterForm(BootstrapMixin, forms.Form):
|
|
model = Cable
|
|
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',
|
|
}
|
|
)
|
|
)
|
|
tenant = FilterChoiceField(
|
|
queryset=Tenant.objects.all(),
|
|
to_field_name='slug',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/tenancy/tenants/",
|
|
value_field='slug',
|
|
filter_for={
|
|
'device_id': 'tenant',
|
|
}
|
|
)
|
|
)
|
|
rack_id = FilterChoiceField(
|
|
queryset=Rack.objects.all(),
|
|
label='Rack',
|
|
null_label='-- None --',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/racks/",
|
|
null_option=True,
|
|
)
|
|
)
|
|
type = forms.MultipleChoiceField(
|
|
choices=add_blank_choice(CableTypeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
status = forms.ChoiceField(
|
|
required=False,
|
|
choices=add_blank_choice(CableStatusChoices),
|
|
widget=StaticSelect2()
|
|
)
|
|
color = forms.CharField(
|
|
max_length=6,
|
|
required=False,
|
|
widget=ColorSelect()
|
|
)
|
|
device_id = FilterChoiceField(
|
|
queryset=Device.objects.all(),
|
|
required=False,
|
|
label='Device',
|
|
widget=APISelectMultiple(
|
|
api_url='/api/dcim/devices/',
|
|
)
|
|
)
|
|
|
|
|
|
#
|
|
# Device bays
|
|
#
|
|
|
|
class DeviceBayFilterForm(DeviceComponentFilterForm):
|
|
model = DeviceBay
|
|
|
|
|
|
class DeviceBayForm(BootstrapMixin, forms.ModelForm):
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = DeviceBay
|
|
fields = [
|
|
'device', 'name', 'description', 'tags',
|
|
]
|
|
widgets = {
|
|
'device': forms.HiddenInput(),
|
|
}
|
|
|
|
|
|
class DeviceBayCreateForm(ComponentForm):
|
|
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=SubdeviceRoleChoices.ROLE_CHILD
|
|
).exclude(pk=device_bay.device.pk)
|
|
|
|
|
|
class DeviceBayCSVForm(forms.ModelForm):
|
|
device = FlexibleModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
to_field_name='name',
|
|
help_text='Name or ID of device',
|
|
error_messages={
|
|
'invalid_choice': 'Device not found.',
|
|
}
|
|
)
|
|
installed_device = FlexibleModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
required=False,
|
|
to_field_name='name',
|
|
help_text='Name or ID of device',
|
|
error_messages={
|
|
'invalid_choice': 'Child device not found.',
|
|
}
|
|
)
|
|
|
|
class Meta:
|
|
model = DeviceBay
|
|
fields = DeviceBay.csv_headers
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
|
|
# Limit installed device choices to devices of the correct type and location
|
|
if self.is_bound:
|
|
try:
|
|
device = self.fields['device'].to_python(self.data['device'])
|
|
except forms.ValidationError:
|
|
device = None
|
|
else:
|
|
try:
|
|
device = self.instance.device
|
|
except Device.DoesNotExist:
|
|
device = None
|
|
|
|
if device:
|
|
self.fields['installed_device'].queryset = Device.objects.filter(
|
|
site=device.site,
|
|
rack=device.rack,
|
|
parent_bay__isnull=True,
|
|
device_type__u_height=0,
|
|
device_type__subdevice_role=SubdeviceRoleChoices.ROLE_CHILD
|
|
).exclude(pk=device.pk)
|
|
else:
|
|
self.fields['installed_device'].queryset = Interface.objects.none()
|
|
|
|
|
|
class DeviceBayBulkRenameForm(BulkRenameForm):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=DeviceBay.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
|
|
|
|
#
|
|
# Connections
|
|
#
|
|
|
|
class ConsoleConnectionFilterForm(BootstrapMixin, forms.Form):
|
|
site = FilterChoiceField(
|
|
queryset=Site.objects.all(),
|
|
to_field_name='slug',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/sites/",
|
|
value_field="slug",
|
|
)
|
|
)
|
|
device_id = FilterChoiceField(
|
|
queryset=Device.objects.all(),
|
|
required=False,
|
|
label='Device',
|
|
widget=APISelectMultiple(
|
|
api_url='/api/dcim/devices/',
|
|
)
|
|
)
|
|
|
|
|
|
class PowerConnectionFilterForm(BootstrapMixin, forms.Form):
|
|
site = FilterChoiceField(
|
|
queryset=Site.objects.all(),
|
|
to_field_name='slug',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/sites/",
|
|
value_field="slug",
|
|
)
|
|
)
|
|
device_id = FilterChoiceField(
|
|
queryset=Device.objects.all(),
|
|
required=False,
|
|
label='Device',
|
|
widget=APISelectMultiple(
|
|
api_url='/api/dcim/devices/',
|
|
)
|
|
)
|
|
|
|
|
|
class InterfaceConnectionFilterForm(BootstrapMixin, forms.Form):
|
|
site = FilterChoiceField(
|
|
queryset=Site.objects.all(),
|
|
to_field_name='slug',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/sites/",
|
|
value_field="slug",
|
|
)
|
|
)
|
|
device_id = FilterChoiceField(
|
|
queryset=Device.objects.all(),
|
|
required=False,
|
|
label='Device',
|
|
widget=APISelectMultiple(
|
|
api_url='/api/dcim/devices/',
|
|
)
|
|
)
|
|
|
|
|
|
#
|
|
# Inventory items
|
|
#
|
|
|
|
class InventoryItemForm(BootstrapMixin, forms.ModelForm):
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = InventoryItem
|
|
fields = [
|
|
'name', 'device', 'manufacturer', 'part_id', 'serial', 'asset_tag', 'description', 'tags',
|
|
]
|
|
widgets = {
|
|
'device': APISelect(
|
|
api_url="/api/dcim/devices/"
|
|
),
|
|
'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):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=InventoryItem.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
device = forms.ModelChoiceField(
|
|
queryset=Device.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/dcim/devices/"
|
|
)
|
|
)
|
|
manufacturer = forms.ModelChoiceField(
|
|
queryset=Manufacturer.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/dcim/manufacturers/"
|
|
)
|
|
)
|
|
part_id = forms.CharField(
|
|
max_length=50,
|
|
required=False,
|
|
label='Part ID'
|
|
)
|
|
description = forms.CharField(
|
|
max_length=100,
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = [
|
|
'manufacturer', 'part_id', 'description',
|
|
]
|
|
|
|
|
|
class InventoryItemFilterForm(BootstrapMixin, forms.Form):
|
|
model = InventoryItem
|
|
q = forms.CharField(
|
|
required=False,
|
|
label='Search'
|
|
)
|
|
region = FilterChoiceField(
|
|
queryset=Region.objects.all(),
|
|
to_field_name='slug',
|
|
required=False,
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/regions/",
|
|
value_field="slug",
|
|
filter_for={
|
|
'site': 'region'
|
|
}
|
|
)
|
|
)
|
|
site = FilterChoiceField(
|
|
queryset=Site.objects.all(),
|
|
to_field_name='slug',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/sites/",
|
|
value_field="slug",
|
|
filter_for={
|
|
'device_id': 'site'
|
|
}
|
|
)
|
|
)
|
|
device_id = FilterChoiceField(
|
|
queryset=Device.objects.all(),
|
|
required=False,
|
|
label='Device',
|
|
widget=APISelect(
|
|
api_url='/api/dcim/devices/',
|
|
)
|
|
)
|
|
manufacturer = FilterChoiceField(
|
|
queryset=Manufacturer.objects.all(),
|
|
to_field_name='slug',
|
|
widget=APISelect(
|
|
api_url="/api/dcim/manufacturers/",
|
|
value_field="slug",
|
|
)
|
|
)
|
|
discovered = forms.NullBooleanField(
|
|
required=False,
|
|
widget=StaticSelect2(
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
)
|
|
)
|
|
|
|
|
|
#
|
|
# Virtual chassis
|
|
#
|
|
|
|
class DeviceSelectionForm(forms.Form):
|
|
pk = forms.ModelMultipleChoiceField(
|
|
queryset=Device.objects.all(),
|
|
widget=forms.MultipleHiddenInput()
|
|
)
|
|
|
|
|
|
class VirtualChassisForm(BootstrapMixin, forms.ModelForm):
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = VirtualChassis
|
|
fields = [
|
|
'master', 'domain', 'tags',
|
|
]
|
|
widgets = {
|
|
'master': SelectWithPK(),
|
|
}
|
|
|
|
|
|
class BaseVCMemberFormSet(forms.BaseModelFormSet):
|
|
|
|
def clean(self):
|
|
super().clean()
|
|
|
|
# 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)
|
|
|
|
|
|
class DeviceVCMembershipForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
model = Device
|
|
fields = [
|
|
'vc_position', 'vc_priority',
|
|
]
|
|
labels = {
|
|
'vc_position': 'Position',
|
|
'vc_priority': 'Priority',
|
|
}
|
|
|
|
def __init__(self, validate_vc_position=False, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
|
|
# 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
|
|
|
|
|
|
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',
|
|
}
|
|
)
|
|
)
|
|
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'
|
|
},
|
|
attrs={
|
|
'nullable': 'true',
|
|
}
|
|
)
|
|
)
|
|
device = ChainedModelChoiceField(
|
|
queryset=Device.objects.filter(
|
|
virtual_chassis__isnull=True
|
|
),
|
|
chains=(
|
|
('site', 'site'),
|
|
('rack', 'rack'),
|
|
),
|
|
label='Device',
|
|
widget=APISelect(
|
|
api_url='/api/dcim/devices/',
|
|
display_field='display_name',
|
|
disabled_indicator='virtual_chassis'
|
|
)
|
|
)
|
|
|
|
def clean_device(self):
|
|
device = self.cleaned_data['device']
|
|
if device.virtual_chassis is not None:
|
|
raise forms.ValidationError(
|
|
"Device {} is already assigned to a virtual chassis.".format(device)
|
|
)
|
|
return device
|
|
|
|
|
|
class VirtualChassisFilterForm(BootstrapMixin, CustomFieldFilterForm):
|
|
model = VirtualChassis
|
|
q = forms.CharField(
|
|
required=False,
|
|
label='Search'
|
|
)
|
|
region = FilterChoiceField(
|
|
queryset=Region.objects.all(),
|
|
to_field_name='slug',
|
|
required=False,
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/regions/",
|
|
value_field="slug",
|
|
filter_for={
|
|
'site': 'region'
|
|
}
|
|
)
|
|
)
|
|
site = FilterChoiceField(
|
|
queryset=Site.objects.all(),
|
|
to_field_name='slug',
|
|
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'
|
|
}
|
|
)
|
|
)
|
|
tenant = FilterChoiceField(
|
|
queryset=Tenant.objects.all(),
|
|
to_field_name='slug',
|
|
null_label='-- None --',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/tenancy/tenants/",
|
|
value_field="slug",
|
|
null_option=True,
|
|
)
|
|
)
|
|
|
|
|
|
#
|
|
# Power panels
|
|
#
|
|
|
|
class PowerPanelForm(BootstrapMixin, forms.ModelForm):
|
|
rack_group = ChainedModelChoiceField(
|
|
queryset=RackGroup.objects.all(),
|
|
chains=(
|
|
('site', 'site'),
|
|
),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url='/api/dcim/rack-groups/',
|
|
)
|
|
)
|
|
|
|
class Meta:
|
|
model = PowerPanel
|
|
fields = [
|
|
'site', 'rack_group', 'name',
|
|
]
|
|
widgets = {
|
|
'site': APISelect(
|
|
api_url="/api/dcim/sites/",
|
|
filter_for={
|
|
'rack_group': 'site_id',
|
|
}
|
|
),
|
|
}
|
|
|
|
|
|
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.',
|
|
}
|
|
)
|
|
rack_group_name = forms.CharField(
|
|
required=False,
|
|
help_text="Rack group name (optional)"
|
|
)
|
|
|
|
class Meta:
|
|
model = PowerPanel
|
|
fields = PowerPanel.csv_headers
|
|
|
|
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)
|
|
)
|
|
|
|
|
|
class PowerPanelFilterForm(BootstrapMixin, CustomFieldFilterForm):
|
|
model = PowerPanel
|
|
q = forms.CharField(
|
|
required=False,
|
|
label='Search'
|
|
)
|
|
region = FilterChoiceField(
|
|
queryset=Region.objects.all(),
|
|
to_field_name='slug',
|
|
required=False,
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/regions/",
|
|
value_field="slug",
|
|
filter_for={
|
|
'site': 'region'
|
|
}
|
|
)
|
|
)
|
|
site = FilterChoiceField(
|
|
queryset=Site.objects.all(),
|
|
to_field_name='slug',
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/sites/",
|
|
value_field="slug",
|
|
filter_for={
|
|
'rack_group_id': 'site',
|
|
}
|
|
)
|
|
)
|
|
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,
|
|
)
|
|
)
|
|
|
|
|
|
#
|
|
# Power feeds
|
|
#
|
|
|
|
class PowerFeedForm(BootstrapMixin, CustomFieldForm):
|
|
site = ChainedModelChoiceField(
|
|
queryset=Site.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url='/api/dcim/sites/',
|
|
filter_for={
|
|
'power_panel': 'site_id',
|
|
'rack': 'site_id',
|
|
}
|
|
)
|
|
)
|
|
comments = CommentField()
|
|
tags = TagField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
model = PowerFeed
|
|
fields = [
|
|
'site', 'power_panel', 'rack', 'name', 'status', 'type', 'supply', 'phase', 'voltage', 'amperage',
|
|
'max_utilization', 'comments', 'tags',
|
|
]
|
|
widgets = {
|
|
'power_panel': APISelect(
|
|
api_url="/api/dcim/power-panels/"
|
|
),
|
|
'rack': APISelect(
|
|
api_url="/api/dcim/racks/"
|
|
),
|
|
'status': StaticSelect2(),
|
|
'type': StaticSelect2(),
|
|
'supply': StaticSelect2(),
|
|
'phase': StaticSelect2(),
|
|
}
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
# Initialize site field
|
|
if self.instance and hasattr(self.instance, 'power_panel'):
|
|
self.initial['site'] = self.instance.power_panel.site
|
|
|
|
|
|
class PowerFeedCSVForm(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.',
|
|
}
|
|
)
|
|
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)"
|
|
)
|
|
status = CSVChoiceField(
|
|
choices=PowerFeedStatusChoices,
|
|
required=False,
|
|
help_text='Operational status'
|
|
)
|
|
type = CSVChoiceField(
|
|
choices=PowerFeedTypeChoices,
|
|
required=False,
|
|
help_text='Primary or redundant'
|
|
)
|
|
supply = CSVChoiceField(
|
|
choices=PowerFeedSupplyChoices,
|
|
required=False,
|
|
help_text='AC/DC'
|
|
)
|
|
phase = CSVChoiceField(
|
|
choices=PowerFeedPhaseChoices,
|
|
required=False,
|
|
help_text='Single or three-phase'
|
|
)
|
|
|
|
class Meta:
|
|
model = PowerFeed
|
|
fields = PowerFeed.csv_headers
|
|
|
|
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)
|
|
except Rack.DoesNotExist:
|
|
raise forms.ValidationError(
|
|
"Rack {} not found in site {}, group {}".format(rack_name, site, rack_group)
|
|
)
|
|
|
|
|
|
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/",
|
|
filter_for={
|
|
'rackgroup': 'site_id',
|
|
}
|
|
)
|
|
)
|
|
rack = forms.ModelChoiceField(
|
|
queryset=Rack.objects.all(),
|
|
required=False,
|
|
widget=APISelect(
|
|
api_url="/api/dcim/racks",
|
|
)
|
|
)
|
|
status = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerFeedStatusChoices),
|
|
required=False,
|
|
initial='',
|
|
widget=StaticSelect2()
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerFeedTypeChoices),
|
|
required=False,
|
|
initial='',
|
|
widget=StaticSelect2()
|
|
)
|
|
supply = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerFeedSupplyChoices),
|
|
required=False,
|
|
initial='',
|
|
widget=StaticSelect2()
|
|
)
|
|
phase = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerFeedPhaseChoices),
|
|
required=False,
|
|
initial='',
|
|
widget=StaticSelect2()
|
|
)
|
|
voltage = forms.IntegerField(
|
|
required=False
|
|
)
|
|
amperage = forms.IntegerField(
|
|
required=False
|
|
)
|
|
max_utilization = forms.IntegerField(
|
|
required=False
|
|
)
|
|
comments = forms.CharField(
|
|
required=False
|
|
)
|
|
|
|
class Meta:
|
|
nullable_fields = [
|
|
'rackgroup', 'comments',
|
|
]
|
|
|
|
|
|
class PowerFeedFilterForm(BootstrapMixin, CustomFieldFilterForm):
|
|
model = PowerFeed
|
|
q = forms.CharField(
|
|
required=False,
|
|
label='Search'
|
|
)
|
|
region = FilterChoiceField(
|
|
queryset=Region.objects.all(),
|
|
to_field_name='slug',
|
|
required=False,
|
|
widget=APISelectMultiple(
|
|
api_url="/api/dcim/regions/",
|
|
value_field="slug",
|
|
filter_for={
|
|
'site': 'region'
|
|
}
|
|
)
|
|
)
|
|
site = 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',
|
|
'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,
|
|
)
|
|
)
|
|
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=PowerFeedStatusChoices,
|
|
required=False,
|
|
widget=StaticSelect2Multiple()
|
|
)
|
|
type = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerFeedTypeChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
supply = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerFeedSupplyChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
phase = forms.ChoiceField(
|
|
choices=add_blank_choice(PowerFeedPhaseChoices),
|
|
required=False,
|
|
widget=StaticSelect2()
|
|
)
|
|
voltage = forms.IntegerField(
|
|
required=False
|
|
)
|
|
amperage = forms.IntegerField(
|
|
required=False
|
|
)
|
|
max_utilization = forms.IntegerField(
|
|
required=False
|
|
)
|