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