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
|
2016-09-20 11:08:25 -04:00
|
|
|
from django.db.models import Count, Q
|
2017-11-07 11:08:23 -05:00
|
|
|
from mptt.forms import TreeNodeChoiceField
|
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
|
|
|
|
2018-07-10 10:00:21 -04:00
|
|
|
from extras.forms import AddRemoveTagsForm, CustomFieldForm, CustomFieldBulkEditForm, CustomFieldFilterForm
|
2017-11-14 15:22:40 -05:00
|
|
|
from ipam.models import IPAddress, VLAN, VLANGroup
|
2017-05-11 17:35:20 -04:00
|
|
|
from tenancy.forms import TenancyForm
|
2016-07-26 16:46:22 -04:00
|
|
|
from tenancy.models import Tenant
|
2016-05-18 16:02:53 -04:00
|
|
|
from utilities.forms import (
|
2018-03-14 14:53:28 -04:00
|
|
|
AnnotatedMultipleChoiceField, APISelect, add_blank_choice, ArrayFieldSelectMultiple, BootstrapMixin, BulkEditForm,
|
2018-10-24 14:59:46 -04:00
|
|
|
BulkEditNullBooleanSelect, ChainedFieldsMixin, ChainedModelChoiceField, ColorSelect, CommentField, ComponentForm,
|
|
|
|
ConfirmationForm, ContentTypeSelect, CSVChoiceField, ExpandableNameField, FilterChoiceField,
|
|
|
|
FilterTreeNodeMultipleChoiceField, FlexibleModelChoiceField, JSONField, Livesearch, SelectWithPK, SmallTextarea,
|
2018-11-15 00:42:01 -05:00
|
|
|
SlugField, BOOLEAN_WITH_BLANK_CHOICES, COLOR_CHOICES,
|
2018-10-24 14:59:46 -04:00
|
|
|
|
2016-05-18 16:02:53 -04:00
|
|
|
)
|
2018-12-07 09:57:55 -05:00
|
|
|
from virtualization.models import Cluster, ClusterGroup
|
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,
|
2018-10-24 13:59:44 -04:00
|
|
|
InventoryItem, Platform, PowerOutlet, PowerOutletTemplate, PowerPort, PowerPortTemplate, Rack, RackGroup,
|
2018-10-25 12:08:13 -04:00
|
|
|
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 />
|
|
|
|
Tagged All: Implies all VLANs are available (w/optional untagged VLAN)
|
|
|
|
"""
|
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
def get_device_by_name_or_pk(name):
|
|
|
|
"""
|
|
|
|
Attempt to retrieve a device by either its name or primary key ('{pk}').
|
|
|
|
"""
|
|
|
|
if re.match(DEVICE_BY_PK_RE, name):
|
|
|
|
pk = name.strip('{}')
|
|
|
|
device = Device.objects.get(pk=pk)
|
|
|
|
else:
|
|
|
|
device = Device.objects.get(name=name)
|
|
|
|
return device
|
|
|
|
|
|
|
|
|
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()
|
|
|
|
|
|
|
|
|
2017-02-28 12:11:43 -05:00
|
|
|
#
|
|
|
|
# Regions
|
|
|
|
#
|
|
|
|
|
|
|
|
class RegionForm(BootstrapMixin, forms.ModelForm):
|
|
|
|
slug = SlugField()
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = Region
|
2018-11-27 11:57:29 -05:00
|
|
|
fields = [
|
|
|
|
'parent', 'name', 'slug',
|
|
|
|
]
|
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
|
|
|
|
#
|
|
|
|
|
2017-05-11 17:35:20 -04:00
|
|
|
class SiteForm(BootstrapMixin, TenancyForm, CustomFieldForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
region = TreeNodeChoiceField(
|
|
|
|
queryset=Region.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
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,
|
|
|
|
}
|
|
|
|
),
|
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
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-06-02 14:49:25 -04:00
|
|
|
class SiteCSVForm(forms.ModelForm):
|
2018-01-25 13:07:04 -05:00
|
|
|
status = CSVChoiceField(
|
2018-03-29 09:50:29 -04:00
|
|
|
choices=SITE_STATUS_CHOICES,
|
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(
|
|
|
|
choices=add_blank_choice(SITE_STATUS_CHOICES),
|
|
|
|
required=False,
|
|
|
|
initial=''
|
|
|
|
)
|
|
|
|
region = TreeNodeChoiceField(
|
|
|
|
queryset=Region.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
tenant = forms.ModelChoiceField(
|
|
|
|
queryset=Tenant.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
asn = forms.IntegerField(
|
|
|
|
min_value=1,
|
|
|
|
max_value=4294967295,
|
|
|
|
required=False,
|
|
|
|
label='ASN'
|
|
|
|
)
|
|
|
|
description = forms.CharField(
|
|
|
|
max_length=100,
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
time_zone = TimeZoneFormField(
|
|
|
|
choices=add_blank_choice(TimeZoneFormField().choices),
|
|
|
|
required=False
|
|
|
|
)
|
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
|
|
|
|
|
|
|
|
2016-08-23 12:05:28 -04:00
|
|
|
class SiteFilterForm(BootstrapMixin, CustomFieldFilterForm):
|
|
|
|
model = Site
|
2018-11-27 11:57:29 -05:00
|
|
|
q = forms.CharField(
|
|
|
|
required=False,
|
|
|
|
label='Search'
|
|
|
|
)
|
2018-03-07 14:16:38 -05:00
|
|
|
status = AnnotatedMultipleChoiceField(
|
|
|
|
choices=SITE_STATUS_CHOICES,
|
|
|
|
annotate=Site.objects.all(),
|
|
|
|
annotate_field='status',
|
|
|
|
required=False
|
|
|
|
)
|
2017-02-28 14:15:15 -05:00
|
|
|
region = FilterTreeNodeMultipleChoiceField(
|
2017-02-28 12:11:43 -05:00
|
|
|
queryset=Region.objects.annotate(filter_count=Count('sites')),
|
|
|
|
to_field_name='slug',
|
2017-02-28 14:15:15 -05:00
|
|
|
required=False,
|
2017-02-28 12:11:43 -05:00
|
|
|
)
|
|
|
|
tenant = FilterChoiceField(
|
|
|
|
queryset=Tenant.objects.annotate(filter_count=Count('sites')),
|
|
|
|
to_field_name='slug',
|
2017-12-26 12:08:22 -05:00
|
|
|
null_label='-- None --'
|
2017-02-28 12:11:43 -05: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):
|
2016-05-20 15:32:17 -04:00
|
|
|
slug = SlugField()
|
2016-05-11 13:30:39 -04:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = RackGroup
|
2018-11-27 11:57:29 -05:00
|
|
|
fields = [
|
|
|
|
'site', 'name', 'slug',
|
|
|
|
]
|
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):
|
2018-11-27 11:57:29 -05:00
|
|
|
site = FilterChoiceField(
|
|
|
|
queryset=Site.objects.annotate(
|
|
|
|
filter_count=Count('rack_groups')
|
|
|
|
),
|
|
|
|
to_field_name='slug'
|
|
|
|
)
|
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 = [
|
|
|
|
'name', 'slug', 'color',
|
|
|
|
]
|
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
|
|
|
|
#
|
|
|
|
|
2017-05-11 17:35:20 -04:00
|
|
|
class RackForm(BootstrapMixin, TenancyForm, CustomFieldForm):
|
2017-05-11 16:24:57 -04:00
|
|
|
group = ChainedModelChoiceField(
|
|
|
|
queryset=RackGroup.objects.all(),
|
2017-05-25 14:33:50 -04:00
|
|
|
chains=(
|
|
|
|
('site', 'site'),
|
|
|
|
),
|
2017-05-11 16:24:57 -04:00
|
|
|
required=False,
|
|
|
|
widget=APISelect(
|
|
|
|
api_url='/api/dcim/rack-groups/?site_id={{site}}',
|
|
|
|
)
|
|
|
|
)
|
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 = {
|
2018-11-27 11:57:29 -05:00
|
|
|
'site': forms.Select(
|
|
|
|
attrs={
|
|
|
|
'filter-for': 'group',
|
|
|
|
}
|
|
|
|
),
|
2016-03-01 11:23:03 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-06-02 14:49:25 -04:00
|
|
|
class RackCSVForm(forms.ModelForm):
|
|
|
|
site = forms.ModelChoiceField(
|
|
|
|
queryset=Site.objects.all(),
|
|
|
|
to_field_name='name',
|
|
|
|
help_text='Name of parent site',
|
|
|
|
error_messages={
|
|
|
|
'invalid_choice': 'Site not found.',
|
|
|
|
}
|
|
|
|
)
|
2017-06-07 14:19:08 -04:00
|
|
|
group_name = forms.CharField(
|
|
|
|
help_text='Name of rack group',
|
|
|
|
required=False
|
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(
|
|
|
|
choices=RACK_STATUS_CHOICES,
|
|
|
|
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(
|
|
|
|
choices=RACK_TYPE_CHOICES,
|
|
|
|
required=False,
|
|
|
|
help_text='Rack type'
|
|
|
|
)
|
|
|
|
width = forms.ChoiceField(
|
2017-06-07 15:56:59 -04:00
|
|
|
choices=(
|
2017-06-07 13:22:06 -04:00
|
|
|
(RACK_WIDTH_19IN, '19'),
|
|
|
|
(RACK_WIDTH_23IN, '23'),
|
|
|
|
),
|
|
|
|
help_text='Rail-to-rail width (in inches)'
|
|
|
|
)
|
2018-11-02 09:51:17 -04:00
|
|
|
outer_unit = CSVChoiceField(
|
|
|
|
choices=RACK_DIMENSION_UNIT_CHOICES,
|
|
|
|
required=False,
|
|
|
|
help_text='Unit for outer dimensions'
|
|
|
|
)
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = Rack
|
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
|
|
|
|
)
|
|
|
|
site = forms.ModelChoiceField(
|
|
|
|
queryset=Site.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
group = forms.ModelChoiceField(
|
|
|
|
queryset=RackGroup.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
tenant = forms.ModelChoiceField(
|
|
|
|
queryset=Tenant.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
status = forms.ChoiceField(
|
|
|
|
choices=add_blank_choice(RACK_STATUS_CHOICES),
|
|
|
|
required=False,
|
|
|
|
initial=''
|
|
|
|
)
|
|
|
|
role = forms.ModelChoiceField(
|
|
|
|
queryset=RackRole.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
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(
|
|
|
|
choices=add_blank_choice(RACK_TYPE_CHOICES),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
width = forms.ChoiceField(
|
|
|
|
choices=add_blank_choice(RACK_WIDTH_CHOICES),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
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(
|
|
|
|
choices=add_blank_choice(RACK_DIMENSION_UNIT_CHOICES),
|
|
|
|
required=False
|
|
|
|
)
|
2018-11-01 16:03:42 -04:00
|
|
|
comments = CommentField(
|
|
|
|
widget=SmallTextarea
|
|
|
|
)
|
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
|
|
|
|
2016-08-23 12:05:28 -04:00
|
|
|
class RackFilterForm(BootstrapMixin, CustomFieldFilterForm):
|
|
|
|
model = Rack
|
2018-11-27 11:57:29 -05:00
|
|
|
q = forms.CharField(
|
|
|
|
required=False,
|
|
|
|
label='Search'
|
|
|
|
)
|
2017-03-01 13:09:19 -05:00
|
|
|
site = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=Site.objects.annotate(
|
|
|
|
filter_count=Count('racks')
|
|
|
|
),
|
2017-03-01 13:09:19 -05:00
|
|
|
to_field_name='slug'
|
|
|
|
)
|
|
|
|
group_id = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=RackGroup.objects.select_related(
|
|
|
|
'site'
|
|
|
|
).annotate(
|
|
|
|
filter_count=Count('racks')
|
|
|
|
),
|
2017-03-01 13:09:19 -05:00
|
|
|
label='Rack group',
|
2017-12-26 12:08:22 -05:00
|
|
|
null_label='-- None --'
|
2017-03-01 13:09:19 -05:00
|
|
|
)
|
|
|
|
tenant = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=Tenant.objects.annotate(
|
|
|
|
filter_count=Count('racks')
|
|
|
|
),
|
2017-03-01 13:09:19 -05:00
|
|
|
to_field_name='slug',
|
2017-12-26 12:08:22 -05:00
|
|
|
null_label='-- None --'
|
2017-03-01 13:09:19 -05:00
|
|
|
)
|
2018-11-01 16:03:42 -04:00
|
|
|
status = AnnotatedMultipleChoiceField(
|
|
|
|
choices=RACK_STATUS_CHOICES,
|
|
|
|
annotate=Rack.objects.all(),
|
|
|
|
annotate_field='status',
|
|
|
|
required=False
|
|
|
|
)
|
2017-03-01 13:09:19 -05:00
|
|
|
role = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=RackRole.objects.annotate(
|
|
|
|
filter_count=Count('racks')
|
|
|
|
),
|
2017-03-01 13:09:19 -05:00
|
|
|
to_field_name='slug',
|
2017-12-26 12:08:22 -05:00
|
|
|
null_label='-- None --'
|
2017-03-01 13:09:19 -05:00
|
|
|
)
|
2016-03-01 11:23:03 -05: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'
|
|
|
|
)
|
|
|
|
)
|
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-04-06 16:26:48 -04:00
|
|
|
class RackReservationFilterForm(BootstrapMixin, forms.Form):
|
2018-11-27 11:57:29 -05:00
|
|
|
q = forms.CharField(
|
|
|
|
required=False,
|
|
|
|
label='Search'
|
|
|
|
)
|
2017-04-06 16:26:48 -04:00
|
|
|
site = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=Site.objects.annotate(
|
|
|
|
filter_count=Count('racks__reservations')
|
|
|
|
),
|
2017-04-06 16:26:48 -04:00
|
|
|
to_field_name='slug'
|
|
|
|
)
|
|
|
|
group_id = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=RackGroup.objects.select_related(
|
|
|
|
'site'
|
|
|
|
).annotate(
|
|
|
|
filter_count=Count('racks__reservations')
|
|
|
|
),
|
2017-04-06 16:26:48 -04:00
|
|
|
label='Rack group',
|
2017-12-26 12:08:22 -05:00
|
|
|
null_label='-- None --'
|
2017-04-06 16:26:48 -04:00
|
|
|
)
|
2017-11-15 12:54:49 -06:00
|
|
|
tenant = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=Tenant.objects.annotate(
|
|
|
|
filter_count=Count('rackreservations')
|
|
|
|
),
|
2017-11-15 12:54:49 -06:00
|
|
|
to_field_name='slug',
|
2018-01-19 10:54:26 -05:00
|
|
|
null_label='-- None --'
|
2017-11-15 12:54:49 -06:00
|
|
|
)
|
2017-04-06 16:26:48 -04:00
|
|
|
|
|
|
|
|
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'
|
|
|
|
),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
tenant = forms.ModelChoiceField(
|
|
|
|
queryset=Tenant.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
description = forms.CharField(
|
|
|
|
max_length=100,
|
|
|
|
required=False
|
|
|
|
)
|
2017-10-31 13:52:35 -04:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
nullable_fields = []
|
|
|
|
|
|
|
|
|
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
|
|
|
|
#
|
|
|
|
|
2016-12-16 10:54:45 -05:00
|
|
|
class DeviceTypeForm(BootstrapMixin, CustomFieldForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
slug = SlugField(
|
|
|
|
slug_source='model'
|
|
|
|
)
|
|
|
|
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 = [
|
2018-11-05 12:02:55 -05:00
|
|
|
'manufacturer', 'model', 'slug', 'part_number', 'u_height', 'is_full_depth', 'subdevice_role', 'comments',
|
|
|
|
'tags',
|
2018-05-08 16:28:26 -04:00
|
|
|
]
|
2017-07-18 04:50:24 +03:00
|
|
|
|
|
|
|
|
|
|
|
class DeviceTypeCSVForm(forms.ModelForm):
|
|
|
|
manufacturer = forms.ModelChoiceField(
|
|
|
|
queryset=Manufacturer.objects.all(),
|
|
|
|
required=True,
|
|
|
|
to_field_name='name',
|
|
|
|
help_text='Manufacturer name',
|
|
|
|
error_messages={
|
|
|
|
'invalid_choice': 'Manufacturer not found.',
|
|
|
|
}
|
|
|
|
)
|
|
|
|
subdevice_role = CSVChoiceField(
|
|
|
|
choices=SUBDEVICE_ROLE_CHOICES,
|
|
|
|
required=False,
|
|
|
|
help_text='Parent/child status'
|
|
|
|
)
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = DeviceType
|
2018-02-02 14:26:16 -05:00
|
|
|
fields = DeviceType.csv_headers
|
2017-07-18 04:50:24 +03:00
|
|
|
help_texts = {
|
|
|
|
'model': 'Model name',
|
|
|
|
'slug': 'URL-friendly slug',
|
|
|
|
}
|
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()
|
|
|
|
)
|
|
|
|
manufacturer = forms.ModelChoiceField(
|
|
|
|
queryset=Manufacturer.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
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'
|
|
|
|
)
|
2017-03-01 13:09:19 -05:00
|
|
|
manufacturer = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=Manufacturer.objects.annotate(
|
|
|
|
filter_count=Count('device_types')
|
|
|
|
),
|
2017-03-01 13:09:19 -05:00
|
|
|
to_field_name='slug'
|
|
|
|
)
|
2018-11-15 00:42:01 -05:00
|
|
|
subdevice_role = forms.NullBooleanField(
|
2018-11-02 10:45:31 -04:00
|
|
|
required=False,
|
2018-11-15 00:42:01 -05:00
|
|
|
label='Subdevice role',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
choices=add_blank_choice(SUBDEVICE_ROLE_CHOICES)
|
|
|
|
)
|
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',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
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',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
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',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
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',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
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',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
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',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
|
|
)
|
2017-03-22 17:29:47 -04: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 = [
|
|
|
|
'device_type', 'name',
|
|
|
|
]
|
2016-12-21 17:20:27 -05:00
|
|
|
widgets = {
|
|
|
|
'device_type': forms.HiddenInput(),
|
|
|
|
}
|
2016-03-04 16:41:24 -05:00
|
|
|
|
|
|
|
|
2017-08-18 13:10:19 -04:00
|
|
|
class ConsolePortTemplateCreateForm(ComponentForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
|
|
|
)
|
2016-03-04 16:41:24 -05:00
|
|
|
|
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 = [
|
|
|
|
'device_type', 'name',
|
|
|
|
]
|
2016-12-21 17:20:27 -05:00
|
|
|
widgets = {
|
|
|
|
'device_type': forms.HiddenInput(),
|
|
|
|
}
|
2016-03-04 16:41:24 -05:00
|
|
|
|
|
|
|
|
2017-08-18 13:10:19 -04:00
|
|
|
class ConsoleServerPortTemplateCreateForm(ComponentForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
|
|
|
)
|
2016-03-04 16:41:24 -05:00
|
|
|
|
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 = [
|
|
|
|
'device_type', 'name',
|
|
|
|
]
|
2016-12-21 17:20:27 -05:00
|
|
|
widgets = {
|
|
|
|
'device_type': forms.HiddenInput(),
|
|
|
|
}
|
2016-03-04 16:41:24 -05:00
|
|
|
|
|
|
|
|
2017-08-18 13:10:19 -04:00
|
|
|
class PowerPortTemplateCreateForm(ComponentForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
|
|
|
)
|
2016-03-04 16:41:24 -05:00
|
|
|
|
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 = [
|
|
|
|
'device_type', 'name',
|
|
|
|
]
|
2016-12-21 17:20:27 -05:00
|
|
|
widgets = {
|
|
|
|
'device_type': forms.HiddenInput(),
|
|
|
|
}
|
2016-03-04 16:41:24 -05:00
|
|
|
|
|
|
|
|
2017-08-18 13:10:19 -04:00
|
|
|
class PowerOutletTemplateCreateForm(ComponentForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
|
|
|
)
|
2016-03-04 16:41:24 -05:00
|
|
|
|
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 = [
|
|
|
|
'device_type', 'name', 'form_factor', 'mgmt_only',
|
|
|
|
]
|
2016-12-21 17:20:27 -05:00
|
|
|
widgets = {
|
|
|
|
'device_type': forms.HiddenInput(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-18 13:10:19 -04:00
|
|
|
class InterfaceTemplateCreateForm(ComponentForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
|
|
|
)
|
|
|
|
form_factor = forms.ChoiceField(
|
|
|
|
choices=IFACE_FF_CHOICES
|
|
|
|
)
|
|
|
|
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()
|
|
|
|
)
|
|
|
|
form_factor = forms.ChoiceField(
|
|
|
|
choices=add_blank_choice(IFACE_FF_CHOICES),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
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(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-10-25 12:08:13 -04:00
|
|
|
class FrontPortTemplateCreateForm(ComponentForm):
|
2018-10-03 14:04:16 -04:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
|
|
|
)
|
|
|
|
type = forms.ChoiceField(
|
2018-10-25 12:08:13 -04:00
|
|
|
choices=PORT_TYPE_CHOICES
|
2018-10-03 14:04:16 -04:00
|
|
|
)
|
|
|
|
rear_port_set = forms.MultipleChoiceField(
|
|
|
|
choices=[],
|
|
|
|
label='Rear ports',
|
|
|
|
help_text='Select one rear port assignment for each front port being created.'
|
|
|
|
)
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
# Determine which rear port positions are occupied. These will be excluded from the list of available mappings.
|
|
|
|
occupied_port_positions = [
|
|
|
|
(front_port.rear_port_id, front_port.rear_port_position)
|
2018-11-02 11:55:52 -04:00
|
|
|
for front_port in self.parent.frontport_templates.all()
|
2018-10-03 14:04:16 -04:00
|
|
|
]
|
|
|
|
|
|
|
|
# Populate rear port choices
|
|
|
|
choices = []
|
2018-11-06 12:43:30 -05:00
|
|
|
rear_ports = RearPortTemplate.objects.filter(device_type=self.parent)
|
2018-10-03 14:04:16 -04:00
|
|
|
for rear_port in rear_ports:
|
|
|
|
for i in range(1, rear_port.positions + 1):
|
|
|
|
if (rear_port.pk, i) not in occupied_port_positions:
|
|
|
|
choices.append(
|
|
|
|
('{}:{}'.format(rear_port.pk, i), '{}:{}'.format(rear_port.name, i))
|
|
|
|
)
|
|
|
|
self.fields['rear_port_set'].choices = choices
|
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
|
|
|
|
# Validate that the number of ports being created equals the number of selected (rear port, position) tuples
|
|
|
|
front_port_count = len(self.cleaned_data['name_pattern'])
|
|
|
|
rear_port_count = len(self.cleaned_data['rear_port_set'])
|
|
|
|
if front_port_count != rear_port_count:
|
|
|
|
raise forms.ValidationError({
|
|
|
|
'rear_port_set': 'The provided name pattern will create {} ports, however {} rear port assignments '
|
|
|
|
'were selected. These counts must match.'.format(front_port_count, rear_port_count)
|
|
|
|
})
|
|
|
|
|
|
|
|
def get_iterative_data(self, iteration):
|
|
|
|
|
|
|
|
# Assign rear port and position from selected set
|
|
|
|
rear_port, position = self.cleaned_data['rear_port_set'][iteration].split(':')
|
|
|
|
|
|
|
|
return {
|
|
|
|
'rear_port': int(rear_port),
|
|
|
|
'rear_port_position': int(position),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-10-25 12:08:13 -04:00
|
|
|
class RearPortTemplateCreateForm(ComponentForm):
|
2018-10-03 14:04:16 -04:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
|
|
|
)
|
|
|
|
type = forms.ChoiceField(
|
2018-10-25 12:08:13 -04:00
|
|
|
choices=PORT_TYPE_CHOICES
|
2018-10-03 14:04:16 -04:00
|
|
|
)
|
|
|
|
positions = forms.IntegerField(
|
|
|
|
min_value=1,
|
|
|
|
max_value=64,
|
|
|
|
initial=1,
|
|
|
|
help_text='The number of front ports which may be mapped to each rear port'
|
|
|
|
)
|
|
|
|
|
|
|
|
|
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(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-18 13:10:19 -04:00
|
|
|
class DeviceBayTemplateCreateForm(ComponentForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
|
|
|
)
|
2016-07-01 17:12:43 -04:00
|
|
|
|
|
|
|
|
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 = [
|
|
|
|
'name', 'slug', 'color', 'vm_role',
|
|
|
|
]
|
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):
|
2016-05-20 15:32:17 -04:00
|
|
|
slug = SlugField()
|
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
|
|
|
|
#
|
|
|
|
|
2017-05-11 17:35:20 -04:00
|
|
|
class DeviceForm(BootstrapMixin, TenancyForm, CustomFieldForm):
|
2017-05-11 16:24:57 -04:00
|
|
|
site = forms.ModelChoiceField(
|
|
|
|
queryset=Site.objects.all(),
|
|
|
|
widget=forms.Select(
|
2018-11-27 11:57:29 -05:00
|
|
|
attrs={
|
|
|
|
'filter-for': 'rack',
|
|
|
|
}
|
2017-05-11 16:24:57 -04:00
|
|
|
)
|
|
|
|
)
|
|
|
|
rack = ChainedModelChoiceField(
|
|
|
|
queryset=Rack.objects.all(),
|
2017-05-25 14:33:50 -04:00
|
|
|
chains=(
|
|
|
|
('site', 'site'),
|
|
|
|
),
|
2017-05-11 16:24:57 -04:00
|
|
|
required=False,
|
|
|
|
widget=APISelect(
|
2017-05-08 13:55:19 -04:00
|
|
|
api_url='/api/dcim/racks/?site_id={{site}}',
|
|
|
|
display_field='display_name',
|
2018-11-27 11:57:29 -05:00
|
|
|
attrs={
|
|
|
|
'filter-for': 'position',
|
|
|
|
}
|
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(
|
|
|
|
api_url='/api/dcim/racks/{{rack}}/units/?face={{face}}',
|
|
|
|
disabled_indicator='device'
|
|
|
|
)
|
2017-05-08 13:55:19 -04:00
|
|
|
)
|
|
|
|
manufacturer = forms.ModelChoiceField(
|
2017-05-11 16:24:57 -04:00
|
|
|
queryset=Manufacturer.objects.all(),
|
|
|
|
widget=forms.Select(
|
2018-11-27 11:57:29 -05:00
|
|
|
attrs={
|
|
|
|
'filter-for': 'device_type',
|
|
|
|
}
|
2017-05-11 16:24:57 -04:00
|
|
|
)
|
2017-05-08 13:55:19 -04:00
|
|
|
)
|
2017-05-11 16:24:57 -04:00
|
|
|
device_type = ChainedModelChoiceField(
|
|
|
|
queryset=DeviceType.objects.all(),
|
2017-05-25 14:33:50 -04:00
|
|
|
chains=(
|
|
|
|
('manufacturer', 'manufacturer'),
|
|
|
|
),
|
2017-05-11 16:24:57 -04:00
|
|
|
label='Device type',
|
|
|
|
widget=APISelect(
|
|
|
|
api_url='/api/dcim/device-types/?manufacturer_id={{manufacturer}}',
|
|
|
|
display_field='model'
|
|
|
|
)
|
2017-05-08 13:55:19 -04:00
|
|
|
)
|
2018-12-07 09:57:55 -05:00
|
|
|
cluster_group = forms.ModelChoiceField(
|
|
|
|
queryset=ClusterGroup.objects.all(),
|
|
|
|
required=False,
|
|
|
|
widget=forms.Select(
|
|
|
|
attrs={'filter-for': 'cluster', 'nullable': 'true'}
|
|
|
|
)
|
|
|
|
)
|
|
|
|
cluster = ChainedModelChoiceField(
|
|
|
|
queryset=Cluster.objects.all(),
|
|
|
|
chains=(
|
|
|
|
('group', 'cluster_group'),
|
|
|
|
),
|
|
|
|
required=False,
|
|
|
|
widget=APISelect(
|
|
|
|
api_url='/api/virtualization/clusters/?group_id={{cluster_group}}',
|
|
|
|
)
|
|
|
|
)
|
2016-03-01 11:23:03 -05:00
|
|
|
comments = CommentField()
|
2018-05-08 16:28:26 -04:00
|
|
|
tags = TagField(required=False)
|
2018-09-18 11:52:12 -04:00
|
|
|
local_context_data = JSONField(required=False)
|
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 = {
|
2018-11-27 11:57:29 -05:00
|
|
|
'face': forms.Select(
|
|
|
|
attrs={
|
|
|
|
'filter-for': 'position',
|
|
|
|
}
|
|
|
|
),
|
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')
|
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'):
|
2017-07-11 14:52:50 -04:00
|
|
|
initial = kwargs.get('initial', {}).copy()
|
2017-05-11 16:24:57 -04:00
|
|
|
initial['manufacturer'] = instance.device_type.manufacturer
|
2017-05-11 17:52:23 -04:00
|
|
|
kwargs['initial'] = initial
|
2016-03-01 11:23:03 -05:00
|
|
|
|
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
|
|
|
|
interface_ips = IPAddress.objects.select_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
|
|
|
|
nat_ips = IPAddress.objects.select_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.
|
|
|
|
self.fields['position'].widget.attrs['api-url'] += '&exclude={}'.format(self.instance.pk)
|
|
|
|
|
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
|
|
|
|
2017-06-02 14:49:25 -04:00
|
|
|
class BaseDeviceCSVForm(forms.ModelForm):
|
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(
|
2018-01-25 12:20:24 -05:00
|
|
|
choices=DEVICE_STATUS_CHOICES,
|
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(
|
|
|
|
choices=RACK_FACE_CHOICES,
|
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
|
|
|
)
|
2018-11-27 11:57:29 -05:00
|
|
|
device_type = forms.ModelChoiceField(
|
|
|
|
queryset=DeviceType.objects.all(),
|
|
|
|
required=False,
|
|
|
|
label='Type'
|
|
|
|
)
|
|
|
|
device_role = forms.ModelChoiceField(
|
|
|
|
queryset=DeviceRole.objects.all(),
|
|
|
|
required=False,
|
|
|
|
label='Role'
|
|
|
|
)
|
|
|
|
tenant = forms.ModelChoiceField(
|
|
|
|
queryset=Tenant.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
platform = forms.ModelChoiceField(
|
|
|
|
queryset=Platform.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
status = forms.ChoiceField(
|
|
|
|
choices=add_blank_choice(DEVICE_STATUS_CHOICES),
|
|
|
|
required=False,
|
|
|
|
initial=''
|
|
|
|
)
|
|
|
|
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
|
|
|
|
2016-08-23 12:05:28 -04:00
|
|
|
class DeviceFilterForm(BootstrapMixin, CustomFieldFilterForm):
|
|
|
|
model = Device
|
2018-11-27 11:57:29 -05:00
|
|
|
q = forms.CharField(
|
|
|
|
required=False,
|
|
|
|
label='Search'
|
|
|
|
)
|
2018-11-06 10:31:56 -05:00
|
|
|
region = FilterTreeNodeMultipleChoiceField(
|
|
|
|
queryset=Region.objects.all(),
|
|
|
|
to_field_name='slug',
|
|
|
|
required=False,
|
|
|
|
)
|
2017-01-24 10:53:59 -05:00
|
|
|
site = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=Site.objects.annotate(
|
|
|
|
filter_count=Count('devices')
|
|
|
|
),
|
2017-01-24 10:53:59 -05:00
|
|
|
to_field_name='slug',
|
|
|
|
)
|
|
|
|
rack_group_id = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=RackGroup.objects.select_related(
|
|
|
|
'site'
|
|
|
|
).annotate(
|
|
|
|
filter_count=Count('racks__devices')
|
|
|
|
),
|
2017-03-01 13:09:19 -05:00
|
|
|
label='Rack group',
|
2017-01-24 10:53:59 -05:00
|
|
|
)
|
2017-05-12 22:41:27 -04:00
|
|
|
rack_id = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=Rack.objects.annotate(
|
|
|
|
filter_count=Count('devices')
|
|
|
|
),
|
2017-05-12 22:41:27 -04:00
|
|
|
label='Rack',
|
2017-12-26 12:08:22 -05:00
|
|
|
null_label='-- None --',
|
2017-05-12 22:41:27 -04:00
|
|
|
)
|
2017-01-24 10:53:59 -05:00
|
|
|
role = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=DeviceRole.objects.annotate(
|
|
|
|
filter_count=Count('devices')
|
|
|
|
),
|
2017-01-24 10:53:59 -05:00
|
|
|
to_field_name='slug',
|
|
|
|
)
|
|
|
|
tenant = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=Tenant.objects.annotate(
|
|
|
|
filter_count=Count('devices')
|
|
|
|
),
|
2017-05-18 14:27:07 -04:00
|
|
|
to_field_name='slug',
|
2017-12-26 12:08:22 -05:00
|
|
|
null_label='-- None --',
|
2017-01-24 10:53:59 -05:00
|
|
|
)
|
2018-11-27 11:57:29 -05:00
|
|
|
manufacturer_id = FilterChoiceField(
|
|
|
|
queryset=Manufacturer.objects.all(),
|
|
|
|
label='Manufacturer'
|
|
|
|
)
|
2017-01-24 10:53:59 -05:00
|
|
|
device_type_id = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=DeviceType.objects.select_related(
|
|
|
|
'manufacturer'
|
|
|
|
).order_by(
|
|
|
|
'model'
|
|
|
|
).annotate(
|
2017-01-24 10:53:59 -05:00
|
|
|
filter_count=Count('instances'),
|
|
|
|
),
|
|
|
|
label='Model',
|
|
|
|
)
|
|
|
|
platform = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=Platform.objects.annotate(
|
|
|
|
filter_count=Count('devices')
|
|
|
|
),
|
2017-01-24 10:53:59 -05:00
|
|
|
to_field_name='slug',
|
2017-12-26 12:08:22 -05:00
|
|
|
null_label='-- None --',
|
2017-01-24 10:53:59 -05:00
|
|
|
)
|
2018-03-07 14:16:38 -05:00
|
|
|
status = AnnotatedMultipleChoiceField(
|
|
|
|
choices=DEVICE_STATUS_CHOICES,
|
|
|
|
annotate=Device.objects.all(),
|
|
|
|
annotate_field='status',
|
|
|
|
required=False
|
|
|
|
)
|
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',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
|
|
)
|
2018-11-15 00:42:01 -05:00
|
|
|
)
|
|
|
|
console_ports = forms.NullBooleanField(
|
|
|
|
required=False,
|
|
|
|
label='Has console ports',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
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',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
|
|
)
|
2018-11-15 00:42:01 -05:00
|
|
|
)
|
|
|
|
power_ports = forms.NullBooleanField(
|
|
|
|
required=False,
|
|
|
|
label='Has power ports',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
|
|
)
|
2018-11-15 00:42:01 -05:00
|
|
|
)
|
|
|
|
power_outlets = forms.NullBooleanField(
|
|
|
|
required=False,
|
|
|
|
label='Has power outlets',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
|
|
)
|
2018-11-15 00:42:01 -05:00
|
|
|
)
|
|
|
|
interfaces = forms.NullBooleanField(
|
|
|
|
required=False,
|
|
|
|
label='Has interfaces',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
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',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.Select(
|
|
|
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
|
|
)
|
2018-02-21 10:55:49 -05: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):
|
2018-11-27 11:57:29 -05:00
|
|
|
form_factor = forms.ChoiceField(
|
|
|
|
choices=IFACE_FF_CHOICES
|
|
|
|
)
|
|
|
|
enabled = forms.BooleanField(
|
|
|
|
required=False,
|
|
|
|
initial=True
|
|
|
|
)
|
|
|
|
mtu = forms.IntegerField(
|
|
|
|
required=False,
|
|
|
|
min_value=1,
|
|
|
|
max_value=32767,
|
|
|
|
label='MTU'
|
|
|
|
)
|
|
|
|
mgmt_only = forms.BooleanField(
|
|
|
|
required=False,
|
|
|
|
label='Management only'
|
|
|
|
)
|
|
|
|
description = forms.CharField(
|
|
|
|
max_length=100,
|
|
|
|
required=False
|
|
|
|
)
|
2016-12-16 16:29:32 -05:00
|
|
|
|
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
#
|
|
|
|
# Console ports
|
|
|
|
#
|
|
|
|
|
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 = [
|
|
|
|
'device', 'name', 'tags',
|
|
|
|
]
|
2016-03-01 11:23:03 -05:00
|
|
|
widgets = {
|
|
|
|
'device': forms.HiddenInput(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-18 13:10:19 -04:00
|
|
|
class ConsolePortCreateForm(ComponentForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
2017-06-05 15:53:03 -04:00
|
|
|
)
|
2018-11-27 11:57:29 -05:00
|
|
|
tags = TagField(
|
|
|
|
required=False
|
2016-03-01 11:23:03 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
# Console server ports
|
|
|
|
#
|
|
|
|
|
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 = [
|
|
|
|
'device', 'name', 'tags',
|
|
|
|
]
|
2016-03-01 11:23:03 -05:00
|
|
|
widgets = {
|
|
|
|
'device': forms.HiddenInput(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-18 13:10:19 -04:00
|
|
|
class ConsoleServerPortCreateForm(ComponentForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
2017-02-17 14:48:00 -05:00
|
|
|
)
|
2018-11-27 11:57:29 -05:00
|
|
|
tags = TagField(
|
|
|
|
required=False
|
2016-03-01 11:23:03 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
#
|
|
|
|
# Power ports
|
|
|
|
#
|
|
|
|
|
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 = [
|
|
|
|
'device', 'name', 'tags',
|
|
|
|
]
|
2016-03-01 11:23:03 -05:00
|
|
|
widgets = {
|
|
|
|
'device': forms.HiddenInput(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-18 13:10:19 -04:00
|
|
|
class PowerPortCreateForm(ComponentForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
2017-02-17 14:48:00 -05:00
|
|
|
)
|
2018-11-27 11:57:29 -05:00
|
|
|
tags = TagField(
|
|
|
|
required=False
|
2016-03-01 11:23:03 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
# Power outlets
|
|
|
|
#
|
|
|
|
|
2016-12-21 14:15:18 -05:00
|
|
|
class PowerOutletForm(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 = PowerOutlet
|
2018-11-27 11:57:29 -05:00
|
|
|
fields = [
|
|
|
|
'device', 'name', 'tags',
|
|
|
|
]
|
2016-03-01 11:23:03 -05:00
|
|
|
widgets = {
|
|
|
|
'device': forms.HiddenInput(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-18 13:10:19 -04:00
|
|
|
class PowerOutletCreateForm(ComponentForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
2017-02-17 14:48:00 -05:00
|
|
|
)
|
2018-11-27 11:57:29 -05:00
|
|
|
tags = TagField(
|
|
|
|
required=False
|
2016-03-01 11:23:03 -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
|
|
|
|
#
|
|
|
|
|
2018-03-07 17:01:51 -05:00
|
|
|
class InterfaceForm(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 = Interface
|
2017-11-14 15:22:40 -05:00
|
|
|
fields = [
|
2018-03-14 14:53:28 -04:00
|
|
|
'device', 'name', 'form_factor', '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(),
|
|
|
|
}
|
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
|
|
|
|
2018-01-30 16:34:42 -05:00
|
|
|
# Limit LAG choices to interfaces belonging to this device (or VC master)
|
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'])
|
2018-11-05 11:51:38 -05:00
|
|
|
self.fields['lag'].queryset = Interface.objects.filter(
|
2018-01-30 16:34:42 -05:00
|
|
|
device__in=[device, device.get_vc_master()], form_factor=IFACE_FF_LAG
|
2017-02-27 16:52:13 -05:00
|
|
|
)
|
|
|
|
else:
|
2018-01-30 16:34:42 -05:00
|
|
|
device = self.instance.device
|
2018-11-05 11:51:38 -05:00
|
|
|
self.fields['lag'].queryset = Interface.objects.filter(
|
2018-01-30 16:34:42 -05:00
|
|
|
device__in=[self.instance.device, self.instance.device.get_vc_master()], form_factor=IFACE_FF_LAG
|
2017-02-27 16:52:13 -05:00
|
|
|
)
|
|
|
|
|
2018-03-07 17:01:51 -05:00
|
|
|
def clean(self):
|
2017-11-14 15:22:40 -05:00
|
|
|
|
2018-11-27 10:52:24 -05:00
|
|
|
super().clean()
|
2018-03-07 17:01:51 -05:00
|
|
|
|
|
|
|
# Validate VLAN assignments
|
|
|
|
tagged_vlans = self.cleaned_data['tagged_vlans']
|
2017-11-14 15:22:40 -05:00
|
|
|
|
2018-03-08 13:25:51 -05:00
|
|
|
# Untagged interfaces cannot be assigned tagged VLANs
|
2018-03-07 17:01:51 -05:00
|
|
|
if self.cleaned_data['mode'] == IFACE_MODE_ACCESS and tagged_vlans:
|
|
|
|
raise forms.ValidationError({
|
|
|
|
'mode': "An access interface cannot have tagged VLANs assigned."
|
|
|
|
})
|
2017-11-14 15:22:40 -05:00
|
|
|
|
2018-03-08 13:25:51 -05:00
|
|
|
# Remove all tagged VLAN assignments from "tagged all" interfaces
|
|
|
|
elif self.cleaned_data['mode'] == IFACE_MODE_TAGGED_ALL:
|
|
|
|
self.cleaned_data['tagged_vlans'] = []
|
|
|
|
|
2018-03-07 17:01:51 -05:00
|
|
|
|
|
|
|
class InterfaceAssignVLANsForm(BootstrapMixin, forms.ModelForm):
|
|
|
|
vlans = forms.MultipleChoiceField(
|
|
|
|
choices=[],
|
|
|
|
label='VLANs',
|
2018-11-27 11:57:29 -05:00
|
|
|
widget=forms.SelectMultiple(
|
|
|
|
attrs={
|
|
|
|
'size': 20,
|
|
|
|
}
|
|
|
|
)
|
2018-03-07 17:01:51 -05:00
|
|
|
)
|
|
|
|
tagged = forms.BooleanField(
|
|
|
|
required=False,
|
|
|
|
initial=True
|
|
|
|
)
|
2017-11-14 15:22:40 -05:00
|
|
|
|
2018-03-07 17:01:51 -05:00
|
|
|
class Meta:
|
|
|
|
model = Interface
|
|
|
|
fields = []
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
|
2018-11-27 10:52:24 -05:00
|
|
|
super().__init__(*args, **kwargs)
|
2018-03-07 17:01:51 -05:00
|
|
|
|
2018-03-08 13:25:51 -05:00
|
|
|
if self.instance.mode == IFACE_MODE_ACCESS:
|
|
|
|
self.initial['tagged'] = False
|
|
|
|
|
|
|
|
# Find all VLANs already assigned to the interface for exclusion from the list
|
|
|
|
assigned_vlans = [v.pk for v in self.instance.tagged_vlans.all()]
|
|
|
|
if self.instance.untagged_vlan is not None:
|
|
|
|
assigned_vlans.append(self.instance.untagged_vlan.pk)
|
|
|
|
|
2018-03-14 14:53:28 -04:00
|
|
|
# Compile VLAN choices
|
|
|
|
vlan_choices = []
|
|
|
|
|
2018-08-09 15:46:18 -04:00
|
|
|
# Add non-grouped global VLANs
|
2018-03-14 14:53:28 -04:00
|
|
|
global_vlans = VLAN.objects.filter(site=None, group=None).exclude(pk__in=assigned_vlans)
|
|
|
|
vlan_choices.append((
|
|
|
|
'Global', [(vlan.pk, vlan) for vlan in global_vlans])
|
|
|
|
)
|
|
|
|
|
|
|
|
# Add grouped global VLANs
|
|
|
|
for group in VLANGroup.objects.filter(site=None):
|
|
|
|
global_group_vlans = VLAN.objects.filter(group=group).exclude(pk__in=assigned_vlans)
|
|
|
|
vlan_choices.append(
|
|
|
|
(group.name, [(vlan.pk, vlan) for vlan in global_group_vlans])
|
|
|
|
)
|
|
|
|
|
2018-08-09 15:46:18 -04:00
|
|
|
site = getattr(self.instance.parent, 'site', None)
|
|
|
|
if site is not None:
|
2018-03-14 14:53:28 -04:00
|
|
|
|
2018-08-09 15:46:18 -04:00
|
|
|
# Add non-grouped site VLANs
|
|
|
|
site_vlans = VLAN.objects.filter(site=site, group=None).exclude(pk__in=assigned_vlans)
|
|
|
|
vlan_choices.append((site.name, [(vlan.pk, vlan) for vlan in site_vlans]))
|
2018-03-14 14:53:28 -04:00
|
|
|
|
|
|
|
# Add grouped site VLANs
|
2018-08-09 15:46:18 -04:00
|
|
|
for group in VLANGroup.objects.filter(site=site):
|
2018-03-14 14:53:28 -04:00
|
|
|
site_group_vlans = VLAN.objects.filter(group=group).exclude(pk__in=assigned_vlans)
|
|
|
|
vlan_choices.append((
|
|
|
|
'{} / {}'.format(group.site.name, group.name),
|
|
|
|
[(vlan.pk, vlan) for vlan in site_group_vlans]
|
|
|
|
))
|
|
|
|
|
2018-03-07 17:01:51 -05:00
|
|
|
self.fields['vlans'].choices = vlan_choices
|
|
|
|
|
2018-03-08 13:25:51 -05:00
|
|
|
def clean(self):
|
|
|
|
|
2018-11-27 10:52:24 -05:00
|
|
|
super().clean()
|
2018-03-08 13:25:51 -05:00
|
|
|
|
|
|
|
# Only untagged VLANs permitted on an access interface
|
|
|
|
if self.instance.mode == IFACE_MODE_ACCESS and len(self.cleaned_data['vlans']) > 1:
|
|
|
|
raise forms.ValidationError("Only one VLAN may be assigned to an access interface.")
|
|
|
|
|
|
|
|
# 'tagged' is required if more than one VLAN is selected
|
|
|
|
if not self.cleaned_data['tagged'] and len(self.cleaned_data['vlans']) > 1:
|
|
|
|
raise forms.ValidationError("Only one untagged VLAN may be selected.")
|
|
|
|
|
2018-03-07 17:01:51 -05:00
|
|
|
def save(self, *args, **kwargs):
|
|
|
|
|
|
|
|
if self.cleaned_data['tagged']:
|
|
|
|
for vlan in self.cleaned_data['vlans']:
|
|
|
|
self.instance.tagged_vlans.add(vlan)
|
|
|
|
else:
|
2018-03-08 13:25:51 -05:00
|
|
|
self.instance.untagged_vlan_id = self.cleaned_data['vlans'][0]
|
2018-03-07 17:01:51 -05:00
|
|
|
|
2018-11-27 10:52:24 -05:00
|
|
|
return super().save(*args, **kwargs)
|
2018-03-07 17:01:51 -05:00
|
|
|
|
|
|
|
|
2018-03-08 13:25:51 -05:00
|
|
|
class InterfaceCreateForm(ComponentForm, forms.Form):
|
2018-11-27 11:57:29 -05:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
|
|
|
)
|
|
|
|
form_factor = forms.ChoiceField(
|
|
|
|
choices=IFACE_FF_CHOICES
|
|
|
|
)
|
|
|
|
enabled = forms.BooleanField(
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
lag = forms.ModelChoiceField(
|
|
|
|
queryset=Interface.objects.all(),
|
|
|
|
required=False,
|
|
|
|
label='Parent LAG'
|
|
|
|
)
|
|
|
|
mtu = forms.IntegerField(
|
|
|
|
required=False,
|
|
|
|
min_value=1,
|
|
|
|
max_value=32767,
|
|
|
|
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(
|
|
|
|
choices=add_blank_choice(IFACE_MODE_CHOICES),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
tags = TagField(
|
|
|
|
required=False
|
|
|
|
)
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2017-02-27 16:52:13 -05:00
|
|
|
def __init__(self, *args, **kwargs):
|
2017-06-23 14:04:15 -04:00
|
|
|
|
|
|
|
# Set interfaces enabled by default
|
2017-07-11 14:52:50 -04:00
|
|
|
kwargs['initial'] = kwargs.get('initial', {}).copy()
|
2017-06-23 14:04:15 -04:00
|
|
|
kwargs['initial'].update({'enabled': True})
|
|
|
|
|
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 belonging to this device (or its VC master)
|
2017-08-18 13:10:19 -04:00
|
|
|
if self.parent is not None:
|
2018-11-05 11:51:38 -05:00
|
|
|
self.fields['lag'].queryset = Interface.objects.filter(
|
2018-01-30 16:34:42 -05:00
|
|
|
device__in=[self.parent, self.parent.get_vc_master()], form_factor=IFACE_FF_LAG
|
2017-02-27 16:52:13 -05:00
|
|
|
)
|
|
|
|
else:
|
|
|
|
self.fields['lag'].queryset = Interface.objects.none()
|
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2018-07-17 10:25:16 -04:00
|
|
|
class InterfaceBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
|
2018-11-27 11:57:29 -05:00
|
|
|
pk = forms.ModelMultipleChoiceField(
|
|
|
|
queryset=Interface.objects.all(),
|
|
|
|
widget=forms.MultipleHiddenInput()
|
|
|
|
)
|
|
|
|
form_factor = forms.ChoiceField(
|
|
|
|
choices=add_blank_choice(IFACE_FF_CHOICES),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
enabled = forms.NullBooleanField(
|
|
|
|
required=False,
|
|
|
|
widget=BulkEditNullBooleanSelect()
|
|
|
|
)
|
|
|
|
lag = forms.ModelChoiceField(
|
|
|
|
queryset=Interface.objects.all(),
|
|
|
|
required=False,
|
|
|
|
label='Parent LAG'
|
|
|
|
)
|
|
|
|
mtu = forms.IntegerField(
|
|
|
|
required=False,
|
|
|
|
min_value=1,
|
|
|
|
max_value=32767,
|
|
|
|
label='MTU'
|
|
|
|
)
|
|
|
|
mgmt_only = forms.NullBooleanField(
|
|
|
|
required=False,
|
|
|
|
widget=BulkEditNullBooleanSelect(),
|
|
|
|
label='Management only'
|
|
|
|
)
|
|
|
|
description = forms.CharField(
|
|
|
|
max_length=100,
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
mode = forms.ChoiceField(
|
|
|
|
choices=add_blank_choice(IFACE_MODE_CHOICES),
|
|
|
|
required=False
|
|
|
|
)
|
2016-10-14 16:38:46 -04:00
|
|
|
|
|
|
|
class Meta:
|
2018-11-27 11:57:29 -05:00
|
|
|
nullable_fields = [
|
|
|
|
'lag', 'mtu', 'description', 'mode',
|
|
|
|
]
|
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)
|
2018-03-01 13:10:36 -05:00
|
|
|
device = self.parent_obj
|
2017-04-12 15:51:14 -04:00
|
|
|
if device is not None:
|
2018-11-05 11:51:38 -05:00
|
|
|
self.fields['lag'].queryset = Interface.objects.filter(
|
|
|
|
device__in=[device, device.get_vc_master()],
|
|
|
|
form_factor=IFACE_FF_LAG
|
2017-02-27 16:52:13 -05:00
|
|
|
)
|
|
|
|
else:
|
|
|
|
self.fields['lag'].choices = []
|
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
|
|
|
#
|
|
|
|
|
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(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-11-20 21:28:19 -05:00
|
|
|
# TODO: Merge with FrontPortTemplateCreateForm to remove duplicate logic
|
2018-10-25 12:08:13 -04:00
|
|
|
class FrontPortCreateForm(ComponentForm):
|
2018-10-03 14:04:16 -04:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
|
|
|
)
|
|
|
|
type = forms.ChoiceField(
|
2018-10-25 12:08:13 -04:00
|
|
|
choices=PORT_TYPE_CHOICES
|
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',
|
|
|
|
help_text='Select one rear port assignment for each front port being created.'
|
|
|
|
)
|
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
|
|
|
|
|
|
|
# Determine which rear port positions are occupied. These will be excluded from the list of available mappings.
|
|
|
|
occupied_port_positions = [
|
|
|
|
(front_port.rear_port_id, front_port.rear_port_position)
|
2018-10-30 14:54:29 -04:00
|
|
|
for front_port in self.parent.frontports.all()
|
2018-10-03 14:04:16 -04:00
|
|
|
]
|
|
|
|
|
|
|
|
# Populate rear port choices
|
|
|
|
choices = []
|
2018-11-06 12:43:30 -05:00
|
|
|
rear_ports = RearPort.objects.filter(device=self.parent)
|
2018-10-03 14:04:16 -04:00
|
|
|
for rear_port in rear_ports:
|
|
|
|
for i in range(1, rear_port.positions + 1):
|
|
|
|
if (rear_port.pk, i) not in occupied_port_positions:
|
|
|
|
choices.append(
|
|
|
|
('{}:{}'.format(rear_port.pk, i), '{}:{}'.format(rear_port.name, i))
|
|
|
|
)
|
|
|
|
self.fields['rear_port_set'].choices = choices
|
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
|
|
|
|
# Validate that the number of ports being created equals the number of selected (rear port, position) tuples
|
|
|
|
front_port_count = len(self.cleaned_data['name_pattern'])
|
|
|
|
rear_port_count = len(self.cleaned_data['rear_port_set'])
|
|
|
|
if front_port_count != rear_port_count:
|
|
|
|
raise forms.ValidationError({
|
|
|
|
'rear_port_set': 'The provided name pattern will create {} ports, however {} rear port assignments '
|
|
|
|
'were selected. These counts must match.'.format(front_port_count, rear_port_count)
|
|
|
|
})
|
|
|
|
|
|
|
|
def get_iterative_data(self, iteration):
|
|
|
|
|
|
|
|
# Assign rear port and position from selected set
|
|
|
|
rear_port, position = self.cleaned_data['rear_port_set'][iteration].split(':')
|
|
|
|
|
|
|
|
return {
|
|
|
|
'rear_port': int(rear_port),
|
|
|
|
'rear_port_position': int(position),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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
|
|
|
#
|
|
|
|
|
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(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-10-25 12:08:13 -04:00
|
|
|
class RearPortCreateForm(ComponentForm):
|
2018-10-03 14:04:16 -04:00
|
|
|
name_pattern = ExpandableNameField(
|
|
|
|
label='Name'
|
|
|
|
)
|
|
|
|
type = forms.ChoiceField(
|
2018-10-25 12:08:13 -04:00
|
|
|
choices=PORT_TYPE_CHOICES
|
2018-10-03 14:04:16 -04:00
|
|
|
)
|
|
|
|
positions = forms.IntegerField(
|
|
|
|
min_value=1,
|
|
|
|
max_value=64,
|
|
|
|
initial=1,
|
|
|
|
help_text='The number of front ports which may be mapped to each rear port'
|
|
|
|
)
|
2018-11-20 21:28:19 -05:00
|
|
|
description = forms.CharField(
|
|
|
|
required=False
|
|
|
|
)
|
2018-10-03 14:04:16 -04:00
|
|
|
|
|
|
|
|
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
|
|
|
|
#
|
|
|
|
|
2018-10-25 15:51:12 -04:00
|
|
|
class CableCreateForm(BootstrapMixin, ChainedFieldsMixin, forms.ModelForm):
|
2018-10-24 14:38:44 -04:00
|
|
|
termination_b_site = forms.ModelChoiceField(
|
2017-02-17 14:48:00 -05:00
|
|
|
queryset=Site.objects.all(),
|
|
|
|
label='Site',
|
|
|
|
required=False,
|
|
|
|
widget=forms.Select(
|
2018-11-27 11:57:29 -05:00
|
|
|
attrs={
|
|
|
|
'filter-for': 'termination_b_rack',
|
|
|
|
}
|
2017-02-17 14:48:00 -05:00
|
|
|
)
|
|
|
|
)
|
2018-10-24 14:38:44 -04:00
|
|
|
termination_b_rack = ChainedModelChoiceField(
|
2017-02-17 14:48:00 -05:00
|
|
|
queryset=Rack.objects.all(),
|
2017-05-25 14:33:50 -04:00
|
|
|
chains=(
|
2018-10-24 14:38:44 -04:00
|
|
|
('site', 'termination_b_site'),
|
2017-05-25 14:33:50 -04:00
|
|
|
),
|
2017-02-17 14:48:00 -05:00
|
|
|
label='Rack',
|
|
|
|
required=False,
|
|
|
|
widget=APISelect(
|
2018-10-24 14:38:44 -04:00
|
|
|
api_url='/api/dcim/racks/?site_id={{termination_b_site}}',
|
2018-11-27 11:57:29 -05:00
|
|
|
attrs={
|
|
|
|
'filter-for': 'termination_b_device',
|
|
|
|
'nullable': 'true',
|
|
|
|
}
|
2017-02-17 14:48:00 -05:00
|
|
|
)
|
|
|
|
)
|
2018-10-24 14:38:44 -04:00
|
|
|
termination_b_device = ChainedModelChoiceField(
|
2017-02-17 14:48:00 -05:00
|
|
|
queryset=Device.objects.all(),
|
2017-05-25 14:33:50 -04:00
|
|
|
chains=(
|
2018-10-24 14:38:44 -04:00
|
|
|
('site', 'termination_b_site'),
|
|
|
|
('rack', 'termination_b_rack'),
|
2017-05-25 14:33:50 -04:00
|
|
|
),
|
2017-02-17 14:48:00 -05:00
|
|
|
label='Device',
|
|
|
|
required=False,
|
|
|
|
widget=APISelect(
|
2018-10-31 15:23:00 -04:00
|
|
|
api_url='/api/dcim/devices/?site_id={{termination_b_site}}&rack_id={{termination_b_rack}}',
|
2017-02-17 14:48:00 -05:00
|
|
|
display_field='display_name',
|
2018-11-27 11:57:29 -05:00
|
|
|
attrs={
|
|
|
|
'filter-for': 'termination_b_id',
|
|
|
|
}
|
2017-02-17 14:48:00 -05:00
|
|
|
)
|
|
|
|
)
|
|
|
|
livesearch = forms.CharField(
|
|
|
|
required=False,
|
|
|
|
label='Device',
|
|
|
|
widget=Livesearch(
|
|
|
|
query_key='q',
|
2017-03-24 17:38:06 -05:00
|
|
|
query_url='dcim-api:device-list',
|
2018-10-24 14:38:44 -04:00
|
|
|
field_to_update='termination_b_device'
|
2017-02-17 14:48:00 -05:00
|
|
|
)
|
|
|
|
)
|
2018-10-24 14:38:44 -04:00
|
|
|
termination_b_type = forms.ModelChoiceField(
|
2018-10-22 16:58:24 -04:00
|
|
|
queryset=ContentType.objects.all(),
|
|
|
|
label='Type',
|
|
|
|
widget=ContentTypeSelect(
|
2018-11-27 11:57:29 -05:00
|
|
|
attrs={
|
|
|
|
'filter-for': 'termination_b_id',
|
|
|
|
}
|
2018-10-22 16:58:24 -04:00
|
|
|
)
|
|
|
|
)
|
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(
|
2018-10-24 14:38:44 -04:00
|
|
|
api_url='/api/dcim/{{termination_b_type}}s/?device_id={{termination_b_device}}',
|
2018-11-14 23:35:15 -05:00
|
|
|
disabled_indicator='cable',
|
|
|
|
url_conditional_append={
|
|
|
|
'termination_b_type__interface': '&type=physical',
|
|
|
|
}
|
2017-02-17 14:48:00 -05:00
|
|
|
)
|
2016-03-01 11:23:03 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
class Meta:
|
2018-10-22 16:58:24 -04:00
|
|
|
model = Cable
|
|
|
|
fields = [
|
2018-10-24 14:38:44 -04:00
|
|
|
'termination_b_site', 'termination_b_rack', 'termination_b_device', 'livesearch', 'termination_b_type',
|
2018-10-31 15:31:44 -04:00
|
|
|
'termination_b_id', 'type', 'status', 'label', 'color', 'length', 'length_unit',
|
2018-10-22 16:58:24 -04:00
|
|
|
]
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2018-10-22 16:58:24 -04:00
|
|
|
def __init__(self, *args, **kwargs):
|
2018-11-27 10:52:24 -05:00
|
|
|
super().__init__(*args, **kwargs)
|
2018-10-22 16:58:24 -04:00
|
|
|
|
|
|
|
# Define available types for endpoint B based on the type of endpoint A
|
2018-10-24 14:38:44 -04:00
|
|
|
termination_a_type = self.instance.termination_a._meta.model_name
|
|
|
|
self.fields['termination_b_type'].queryset = ContentType.objects.filter(
|
|
|
|
model__in=COMPATIBLE_TERMINATION_TYPES.get(termination_a_type)
|
2018-11-14 12:17:18 -05:00
|
|
|
).exclude(
|
|
|
|
model='circuittermination'
|
2018-10-22 16:58:24 -04:00
|
|
|
)
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
|
2018-10-25 15:51:12 -04:00
|
|
|
class CableForm(BootstrapMixin, forms.ModelForm):
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = Cable
|
2018-11-27 11:57:29 -05:00
|
|
|
fields = [
|
|
|
|
'type', 'status', 'label', 'color', 'length', 'length_unit',
|
2016-03-01 11:23:03 -05:00
|
|
|
]
|
|
|
|
|
|
|
|
|
2018-10-31 17:05:25 -04:00
|
|
|
class CableCSVForm(forms.ModelForm):
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2018-10-31 17:05:25 -04:00
|
|
|
# Termination A
|
|
|
|
side_a_device = FlexibleModelChoiceField(
|
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(),
|
2018-11-27 11:57:29 -05:00
|
|
|
limit_choices_to={
|
|
|
|
'model__in': CABLE_TERMINATION_TYPES,
|
|
|
|
},
|
2018-11-01 09:59:53 -04:00
|
|
|
to_field_name='model',
|
|
|
|
help_text='Side A type'
|
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(),
|
2018-11-27 11:57:29 -05:00
|
|
|
limit_choices_to={
|
|
|
|
'model__in': CABLE_TERMINATION_TYPES,
|
|
|
|
},
|
2018-11-01 09:59:53 -04:00
|
|
|
to_field_name='model',
|
|
|
|
help_text='Side B type'
|
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(
|
2017-06-07 13:22:06 -04:00
|
|
|
choices=CONNECTION_STATUS_CHOICES,
|
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(
|
|
|
|
choices=CABLE_TYPE_CHOICES,
|
|
|
|
required=False,
|
|
|
|
help_text='Cable type'
|
|
|
|
)
|
|
|
|
length_unit = CSVChoiceField(
|
2018-11-02 09:51:17 -04:00
|
|
|
choices=CABLE_LENGTH_UNIT_CHOICES,
|
2018-11-01 09:59:53 -04:00
|
|
|
required=False,
|
|
|
|
help_text='Length unit'
|
|
|
|
)
|
2016-03-01 11:23:03 -05:00
|
|
|
|
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(
|
|
|
|
choices=add_blank_choice(CABLE_TYPE_CHOICES),
|
|
|
|
required=False,
|
|
|
|
initial=''
|
|
|
|
)
|
|
|
|
status = forms.ChoiceField(
|
|
|
|
choices=add_blank_choice(CONNECTION_STATUS_CHOICES),
|
|
|
|
required=False,
|
|
|
|
initial=''
|
|
|
|
)
|
|
|
|
label = forms.CharField(
|
|
|
|
max_length=100,
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
color = forms.CharField(
|
|
|
|
max_length=6,
|
|
|
|
required=False,
|
|
|
|
widget=ColorSelect()
|
|
|
|
)
|
|
|
|
length = forms.IntegerField(
|
|
|
|
min_value=1,
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
length_unit = forms.ChoiceField(
|
2018-11-02 09:51:17 -04:00
|
|
|
choices=add_blank_choice(CABLE_LENGTH_UNIT_CHOICES),
|
2018-11-01 13:19:24 -04:00
|
|
|
required=False,
|
|
|
|
initial=''
|
|
|
|
)
|
|
|
|
|
|
|
|
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'
|
|
|
|
)
|
2018-10-24 14:59:46 -04:00
|
|
|
type = AnnotatedMultipleChoiceField(
|
|
|
|
choices=CABLE_TYPE_CHOICES,
|
|
|
|
annotate=Cable.objects.all(),
|
|
|
|
annotate_field='type',
|
|
|
|
required=False
|
|
|
|
)
|
2018-11-28 14:22:55 -05:00
|
|
|
color = AnnotatedMultipleChoiceField(
|
|
|
|
choices=COLOR_CHOICES,
|
|
|
|
annotate=Cable.objects.all(),
|
|
|
|
annotate_field='color',
|
2018-10-24 14:59:46 -04:00
|
|
|
required=False
|
|
|
|
)
|
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
|
|
|
|
#
|
|
|
|
|
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 = [
|
|
|
|
'device', 'name', 'tags',
|
|
|
|
]
|
2016-07-01 17:12:43 -04:00
|
|
|
widgets = {
|
|
|
|
'device': forms.HiddenInput(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-18 13:10:19 -04:00
|
|
|
class DeviceBayCreateForm(ComponentForm):
|
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',
|
|
|
|
help_text="Child devices must first be created and assigned to the site/rack of the parent device."
|
|
|
|
)
|
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,
|
|
|
|
device_type__subdevice_role=SUBDEVICE_ROLE_CHILD
|
|
|
|
).exclude(pk=device_bay.device.pk)
|
2016-07-01 17:12:43 -04:00
|
|
|
|
|
|
|
|
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):
|
2018-11-27 11:57:29 -05:00
|
|
|
site = forms.ModelChoiceField(
|
|
|
|
queryset=Site.objects.all(),
|
|
|
|
required=False,
|
|
|
|
to_field_name='slug'
|
|
|
|
)
|
|
|
|
device = forms.CharField(
|
|
|
|
required=False,
|
|
|
|
label='Device name'
|
|
|
|
)
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
|
2016-12-21 14:15:18 -05:00
|
|
|
class PowerConnectionFilterForm(BootstrapMixin, forms.Form):
|
2018-11-27 11:57:29 -05:00
|
|
|
site = forms.ModelChoiceField(
|
|
|
|
queryset=Site.objects.all(),
|
|
|
|
required=False,
|
|
|
|
to_field_name='slug'
|
|
|
|
)
|
|
|
|
device = forms.CharField(
|
|
|
|
required=False,
|
|
|
|
label='Device name'
|
|
|
|
)
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
|
2016-12-21 14:15:18 -05:00
|
|
|
class InterfaceConnectionFilterForm(BootstrapMixin, forms.Form):
|
2018-11-27 11:57:29 -05:00
|
|
|
site = forms.ModelChoiceField(
|
|
|
|
queryset=Site.objects.all(),
|
|
|
|
required=False,
|
|
|
|
to_field_name='slug'
|
|
|
|
)
|
|
|
|
device = forms.CharField(
|
|
|
|
required=False,
|
|
|
|
label='Device name'
|
|
|
|
)
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
|
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):
|
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 = [
|
|
|
|
'name', 'manufacturer', 'part_id', 'serial', 'asset_tag', 'description', 'tags',
|
|
|
|
]
|
2017-11-17 16:47:26 -05:00
|
|
|
|
|
|
|
|
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()
|
|
|
|
)
|
|
|
|
manufacturer = forms.ModelChoiceField(
|
|
|
|
queryset=Manufacturer.objects.all(),
|
|
|
|
required=False
|
|
|
|
)
|
|
|
|
part_id = forms.CharField(
|
|
|
|
max_length=50,
|
|
|
|
required=False,
|
|
|
|
label='Part ID'
|
|
|
|
)
|
|
|
|
description = forms.CharField(
|
|
|
|
max_length=100,
|
|
|
|
required=False
|
|
|
|
)
|
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'
|
|
|
|
)
|
|
|
|
device = forms.CharField(
|
|
|
|
required=False,
|
|
|
|
label='Device name'
|
|
|
|
)
|
2018-01-30 17:46:00 -05:00
|
|
|
manufacturer = FilterChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=Manufacturer.objects.annotate(
|
|
|
|
filter_count=Count('inventory_items')
|
|
|
|
),
|
2018-01-30 17:46:00 -05:00
|
|
|
to_field_name='slug',
|
|
|
|
null_label='-- None --'
|
|
|
|
)
|
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
|
|
|
|
|
|
|
|
|
2018-02-01 11:39:13 -05:00
|
|
|
class VCMemberSelectForm(BootstrapMixin, ChainedFieldsMixin, forms.Form):
|
|
|
|
site = forms.ModelChoiceField(
|
|
|
|
queryset=Site.objects.all(),
|
|
|
|
label='Site',
|
|
|
|
required=False,
|
|
|
|
widget=forms.Select(
|
2018-11-27 11:57:29 -05:00
|
|
|
attrs={
|
|
|
|
'filter-for': 'rack',
|
|
|
|
}
|
2018-02-01 11:39:13 -05:00
|
|
|
)
|
|
|
|
)
|
|
|
|
rack = ChainedModelChoiceField(
|
|
|
|
queryset=Rack.objects.all(),
|
|
|
|
chains=(
|
|
|
|
('site', 'site'),
|
|
|
|
),
|
|
|
|
label='Rack',
|
|
|
|
required=False,
|
|
|
|
widget=APISelect(
|
|
|
|
api_url='/api/dcim/racks/?site_id={{site}}',
|
2018-11-27 11:57:29 -05:00
|
|
|
attrs={
|
|
|
|
'filter-for': 'device',
|
|
|
|
'nullable': 'true',
|
|
|
|
}
|
2018-02-01 11:39:13 -05:00
|
|
|
)
|
|
|
|
)
|
|
|
|
device = ChainedModelChoiceField(
|
2018-11-27 11:57:29 -05:00
|
|
|
queryset=Device.objects.filter(
|
|
|
|
virtual_chassis__isnull=True
|
|
|
|
),
|
2018-02-01 11:39:13 -05:00
|
|
|
chains=(
|
|
|
|
('site', 'site'),
|
|
|
|
('rack', 'rack'),
|
|
|
|
),
|
|
|
|
label='Device',
|
|
|
|
widget=APISelect(
|
|
|
|
api_url='/api/dcim/devices/?site_id={{site}}&rack_id={{rack}}',
|
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'
|
|
|
|
)
|
2018-02-21 09:53:23 -05:00
|
|
|
site = FilterChoiceField(
|
|
|
|
queryset=Site.objects.all(),
|
|
|
|
to_field_name='slug',
|
|
|
|
)
|
|
|
|
tenant = FilterChoiceField(
|
|
|
|
queryset=Tenant.objects.all(),
|
|
|
|
to_field_name='slug',
|
|
|
|
null_label='-- None --',
|
|
|
|
)
|