forms.py 52 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242
  1. import re
  2. from django import forms
  3. from django.core.exceptions import ValidationError
  4. from django.db.models import Count, Q
  5. from extras.forms import CustomFieldForm, CustomFieldBulkEditForm, CustomFieldFilterForm
  6. from ipam.models import IPAddress
  7. from tenancy.models import Tenant
  8. from utilities.forms import (
  9. APISelect, add_blank_choice, BootstrapMixin, BulkEditForm, BulkImportForm, CommentField, CSVDataField,
  10. ExpandableNameField, FilterChoiceField, FlexibleModelChoiceField, Livesearch, SelectWithDisabled, SmallTextarea,
  11. SlugField,
  12. )
  13. from .models import (
  14. DeviceBay, DeviceBayTemplate, CONNECTION_STATUS_CHOICES, CONNECTION_STATUS_PLANNED, CONNECTION_STATUS_CONNECTED,
  15. ConsolePort, ConsolePortTemplate, ConsoleServerPort, ConsoleServerPortTemplate, Device, DeviceRole, DeviceType,
  16. Interface, IFACE_FF_VIRTUAL, InterfaceConnection, InterfaceTemplate, Manufacturer, Module, Platform, PowerOutlet,
  17. PowerOutletTemplate, PowerPort, PowerPortTemplate, RACK_TYPE_CHOICES, RACK_WIDTH_CHOICES, Rack, RackGroup, RackRole,
  18. Site, STATUS_CHOICES, SUBDEVICE_ROLE_CHILD
  19. )
  20. FORM_STATUS_CHOICES = [
  21. ['', '---------'],
  22. ]
  23. FORM_STATUS_CHOICES += STATUS_CHOICES
  24. DEVICE_BY_PK_RE = '{\d+\}'
  25. def get_device_by_name_or_pk(name):
  26. """
  27. Attempt to retrieve a device by either its name or primary key ('{pk}').
  28. """
  29. if re.match(DEVICE_BY_PK_RE, name):
  30. pk = name.strip('{}')
  31. device = Device.objects.get(pk=pk)
  32. else:
  33. device = Device.objects.get(name=name)
  34. return device
  35. def validate_connection_status(value):
  36. """
  37. Custom validator for connection statuses. value must be either "planned" or "connected" (case-insensitive).
  38. """
  39. if value.lower() not in ['planned', 'connected']:
  40. raise ValidationError('Invalid connection status ({}); must be either "planned" or "connected".'.format(value))
  41. #
  42. # Sites
  43. #
  44. class SiteForm(BootstrapMixin, CustomFieldForm):
  45. slug = SlugField()
  46. comments = CommentField()
  47. class Meta:
  48. model = Site
  49. fields = ['name', 'slug', 'tenant', 'facility', 'asn', 'physical_address', 'shipping_address', 'comments']
  50. widgets = {
  51. 'physical_address': SmallTextarea(attrs={'rows': 3}),
  52. 'shipping_address': SmallTextarea(attrs={'rows': 3}),
  53. }
  54. help_texts = {
  55. 'name': "Full name of the site",
  56. 'facility': "Data center provider and facility (e.g. Equinix NY7)",
  57. 'asn': "BGP autonomous system number",
  58. 'physical_address': "Physical location of the building (e.g. for GPS)",
  59. 'shipping_address': "If different from the physical address"
  60. }
  61. class SiteFromCSVForm(forms.ModelForm):
  62. tenant = forms.ModelChoiceField(Tenant.objects.all(), to_field_name='name', required=False,
  63. error_messages={'invalid_choice': 'Tenant not found.'})
  64. class Meta:
  65. model = Site
  66. fields = ['name', 'slug', 'tenant', 'facility', 'asn']
  67. class SiteImportForm(BulkImportForm, BootstrapMixin):
  68. csv = CSVDataField(csv_form=SiteFromCSVForm)
  69. class SiteBulkEditForm(BootstrapMixin, CustomFieldBulkEditForm):
  70. pk = forms.ModelMultipleChoiceField(queryset=Site.objects.all(), widget=forms.MultipleHiddenInput)
  71. tenant = forms.ModelChoiceField(queryset=Tenant.objects.all(), required=False)
  72. class Meta:
  73. nullable_fields = ['tenant']
  74. class SiteFilterForm(BootstrapMixin, CustomFieldFilterForm):
  75. model = Site
  76. tenant = FilterChoiceField(queryset=Tenant.objects.annotate(filter_count=Count('sites')), to_field_name='slug',
  77. null_option=(0, 'None'))
  78. #
  79. # Rack groups
  80. #
  81. class RackGroupForm(forms.ModelForm, BootstrapMixin):
  82. slug = SlugField()
  83. class Meta:
  84. model = RackGroup
  85. fields = ['site', 'name', 'slug']
  86. class RackGroupFilterForm(forms.Form, BootstrapMixin):
  87. site = FilterChoiceField(queryset=Site.objects.annotate(filter_count=Count('rack_groups')), to_field_name='slug')
  88. #
  89. # Rack roles
  90. #
  91. class RackRoleForm(forms.ModelForm, BootstrapMixin):
  92. slug = SlugField()
  93. class Meta:
  94. model = RackRole
  95. fields = ['name', 'slug', 'color']
  96. #
  97. # Racks
  98. #
  99. class RackForm(BootstrapMixin, CustomFieldForm):
  100. group = forms.ModelChoiceField(queryset=RackGroup.objects.all(), required=False, label='Group', widget=APISelect(
  101. api_url='/api/dcim/rack-groups/?site_id={{site}}',
  102. ))
  103. comments = CommentField()
  104. class Meta:
  105. model = Rack
  106. fields = ['site', 'group', 'name', 'facility_id', 'tenant', 'role', 'type', 'width', 'u_height', 'comments']
  107. help_texts = {
  108. 'site': "The site at which the rack exists",
  109. 'name': "Organizational rack name",
  110. 'facility_id': "The unique rack ID assigned by the facility",
  111. 'u_height': "Height in rack units",
  112. }
  113. widgets = {
  114. 'site': forms.Select(attrs={'filter-for': 'group'}),
  115. }
  116. def __init__(self, *args, **kwargs):
  117. super(RackForm, self).__init__(*args, **kwargs)
  118. # Limit rack group choices
  119. if self.is_bound and self.data.get('site'):
  120. self.fields['group'].queryset = RackGroup.objects.filter(site__pk=self.data['site'])
  121. elif self.initial.get('site'):
  122. self.fields['group'].queryset = RackGroup.objects.filter(site=self.initial['site'])
  123. else:
  124. self.fields['group'].choices = []
  125. class RackFromCSVForm(forms.ModelForm):
  126. site = forms.ModelChoiceField(queryset=Site.objects.all(), to_field_name='name',
  127. error_messages={'invalid_choice': 'Site not found.'})
  128. group_name = forms.CharField(required=False)
  129. tenant = forms.ModelChoiceField(Tenant.objects.all(), to_field_name='name', required=False,
  130. error_messages={'invalid_choice': 'Tenant not found.'})
  131. role = forms.ModelChoiceField(RackRole.objects.all(), to_field_name='name', required=False,
  132. error_messages={'invalid_choice': 'Role not found.'})
  133. type = forms.CharField(required=False)
  134. class Meta:
  135. model = Rack
  136. fields = ['site', 'group_name', 'name', 'facility_id', 'tenant', 'role', 'type', 'width', 'u_height']
  137. def clean(self):
  138. site = self.cleaned_data.get('site')
  139. group = self.cleaned_data.get('group_name')
  140. # Validate rack group
  141. if site and group:
  142. try:
  143. self.instance.group = RackGroup.objects.get(site=site, name=group)
  144. except RackGroup.DoesNotExist:
  145. self.add_error('group_name', "Invalid rack group ({})".format(group))
  146. def clean_type(self):
  147. rack_type = self.cleaned_data['type']
  148. if not rack_type:
  149. return None
  150. try:
  151. choices = {v.lower(): k for k, v in RACK_TYPE_CHOICES}
  152. return choices[rack_type.lower()]
  153. except KeyError:
  154. raise forms.ValidationError('Invalid rack type ({}). Valid choices are: {}.'.format(
  155. rack_type,
  156. ', '.join({v: k for k, v in RACK_TYPE_CHOICES}),
  157. ))
  158. class RackImportForm(BulkImportForm, BootstrapMixin):
  159. csv = CSVDataField(csv_form=RackFromCSVForm)
  160. class RackBulkEditForm(BootstrapMixin, CustomFieldBulkEditForm):
  161. pk = forms.ModelMultipleChoiceField(queryset=Rack.objects.all(), widget=forms.MultipleHiddenInput)
  162. site = forms.ModelChoiceField(queryset=Site.objects.all(), required=False, label='Site')
  163. group = forms.ModelChoiceField(queryset=RackGroup.objects.all(), required=False, label='Group')
  164. tenant = forms.ModelChoiceField(queryset=Tenant.objects.all(), required=False)
  165. role = forms.ModelChoiceField(queryset=RackRole.objects.all(), required=False)
  166. type = forms.ChoiceField(choices=add_blank_choice(RACK_TYPE_CHOICES), required=False, label='Type')
  167. width = forms.ChoiceField(choices=add_blank_choice(RACK_WIDTH_CHOICES), required=False, label='Width')
  168. u_height = forms.IntegerField(required=False, label='Height (U)')
  169. comments = CommentField()
  170. class Meta:
  171. nullable_fields = ['group', 'tenant', 'role', 'comments']
  172. class RackFilterForm(BootstrapMixin, CustomFieldFilterForm):
  173. model = Rack
  174. site = FilterChoiceField(queryset=Site.objects.annotate(filter_count=Count('racks')), to_field_name='slug')
  175. group_id = FilterChoiceField(queryset=RackGroup.objects.select_related('site')
  176. .annotate(filter_count=Count('racks')), label='Rack group', null_option=(0, 'None'))
  177. tenant = FilterChoiceField(queryset=Tenant.objects.annotate(filter_count=Count('racks')), to_field_name='slug',
  178. null_option=(0, 'None'))
  179. role = FilterChoiceField(queryset=RackRole.objects.annotate(filter_count=Count('racks')), to_field_name='slug',
  180. null_option=(0, 'None'))
  181. #
  182. # Manufacturers
  183. #
  184. class ManufacturerForm(forms.ModelForm, BootstrapMixin):
  185. slug = SlugField()
  186. class Meta:
  187. model = Manufacturer
  188. fields = ['name', 'slug']
  189. #
  190. # Device types
  191. #
  192. class DeviceTypeForm(forms.ModelForm, BootstrapMixin):
  193. slug = SlugField(slug_source='model')
  194. class Meta:
  195. model = DeviceType
  196. fields = ['manufacturer', 'model', 'slug', 'part_number', 'u_height', 'is_full_depth', 'is_console_server',
  197. 'is_pdu', 'is_network_device', 'subdevice_role']
  198. class DeviceTypeBulkEditForm(BulkEditForm, BootstrapMixin):
  199. pk = forms.ModelMultipleChoiceField(queryset=DeviceType.objects.all(), widget=forms.MultipleHiddenInput)
  200. manufacturer = forms.ModelChoiceField(queryset=Manufacturer.objects.all(), required=False)
  201. u_height = forms.IntegerField(min_value=1, required=False)
  202. class DeviceTypeFilterForm(forms.Form, BootstrapMixin):
  203. manufacturer = FilterChoiceField(queryset=Manufacturer.objects.annotate(filter_count=Count('device_types')),
  204. to_field_name='slug')
  205. #
  206. # Device component templates
  207. #
  208. class ConsolePortTemplateForm(forms.ModelForm, BootstrapMixin):
  209. name_pattern = ExpandableNameField(label='Name')
  210. class Meta:
  211. model = ConsolePortTemplate
  212. fields = ['name_pattern']
  213. class ConsoleServerPortTemplateForm(forms.ModelForm, BootstrapMixin):
  214. name_pattern = ExpandableNameField(label='Name')
  215. class Meta:
  216. model = ConsoleServerPortTemplate
  217. fields = ['name_pattern']
  218. class PowerPortTemplateForm(forms.ModelForm, BootstrapMixin):
  219. name_pattern = ExpandableNameField(label='Name')
  220. class Meta:
  221. model = PowerPortTemplate
  222. fields = ['name_pattern']
  223. class PowerOutletTemplateForm(forms.ModelForm, BootstrapMixin):
  224. name_pattern = ExpandableNameField(label='Name')
  225. class Meta:
  226. model = PowerOutletTemplate
  227. fields = ['name_pattern']
  228. class InterfaceTemplateForm(forms.ModelForm, BootstrapMixin):
  229. name_pattern = ExpandableNameField(label='Name')
  230. class Meta:
  231. model = InterfaceTemplate
  232. fields = ['name_pattern', 'form_factor', 'mgmt_only']
  233. class DeviceBayTemplateForm(forms.ModelForm, BootstrapMixin):
  234. name_pattern = ExpandableNameField(label='Name')
  235. class Meta:
  236. model = DeviceBayTemplate
  237. fields = ['name_pattern']
  238. #
  239. # Device roles
  240. #
  241. class DeviceRoleForm(forms.ModelForm, BootstrapMixin):
  242. slug = SlugField()
  243. class Meta:
  244. model = DeviceRole
  245. fields = ['name', 'slug', 'color']
  246. #
  247. # Platforms
  248. #
  249. class PlatformForm(forms.ModelForm, BootstrapMixin):
  250. slug = SlugField()
  251. class Meta:
  252. model = Platform
  253. fields = ['name', 'slug']
  254. #
  255. # Devices
  256. #
  257. class DeviceForm(BootstrapMixin, CustomFieldForm):
  258. site = forms.ModelChoiceField(queryset=Site.objects.all(), widget=forms.Select(attrs={'filter-for': 'rack'}))
  259. rack = forms.ModelChoiceField(queryset=Rack.objects.all(), widget=APISelect(
  260. api_url='/api/dcim/racks/?site_id={{site}}',
  261. display_field='display_name',
  262. attrs={'filter-for': 'position'}
  263. ))
  264. position = forms.TypedChoiceField(required=False, empty_value=None,
  265. help_text="For multi-U devices, this is the lowest occupied rack unit.",
  266. widget=APISelect(api_url='/api/dcim/racks/{{rack}}/rack-units/?face={{face}}',
  267. disabled_indicator='device'))
  268. manufacturer = forms.ModelChoiceField(queryset=Manufacturer.objects.all(),
  269. widget=forms.Select(attrs={'filter-for': 'device_type'}))
  270. device_type = forms.ModelChoiceField(queryset=DeviceType.objects.all(), label='Device type', widget=APISelect(
  271. api_url='/api/dcim/device-types/?manufacturer_id={{manufacturer}}',
  272. display_field='model'
  273. ))
  274. comments = CommentField()
  275. class Meta:
  276. model = Device
  277. fields = ['name', 'device_role', 'tenant', 'device_type', 'serial', 'asset_tag', 'site', 'rack', 'position',
  278. 'face', 'status', 'platform', 'primary_ip4', 'primary_ip6', 'comments']
  279. help_texts = {
  280. 'device_role': "The function this device serves",
  281. 'serial': "Chassis serial number",
  282. }
  283. widgets = {
  284. 'face': forms.Select(attrs={'filter-for': 'position'}),
  285. 'manufacturer': forms.Select(attrs={'filter-for': 'device_type'}),
  286. }
  287. def __init__(self, *args, **kwargs):
  288. super(DeviceForm, self).__init__(*args, **kwargs)
  289. if self.instance.pk:
  290. # Initialize helper selections
  291. self.initial['site'] = self.instance.rack.site
  292. self.initial['manufacturer'] = self.instance.device_type.manufacturer
  293. # Compile list of choices for primary IPv4 and IPv6 addresses
  294. for family in [4, 6]:
  295. ip_choices = []
  296. interface_ips = IPAddress.objects.filter(family=family, interface__device=self.instance)
  297. ip_choices += [(ip.id, u'{} ({})'.format(ip.address, ip.interface)) for ip in interface_ips]
  298. nat_ips = IPAddress.objects.filter(family=family, nat_inside__interface__device=self.instance)\
  299. .select_related('nat_inside__interface')
  300. ip_choices += [(ip.id, u'{} ({} NAT)'.format(ip.address, ip.nat_inside.interface)) for ip in nat_ips]
  301. self.fields['primary_ip{}'.format(family)].choices = [(None, '---------')] + ip_choices
  302. else:
  303. # An object that doesn't exist yet can't have any IPs assigned to it
  304. self.fields['primary_ip4'].choices = []
  305. self.fields['primary_ip4'].widget.attrs['readonly'] = True
  306. self.fields['primary_ip6'].choices = []
  307. self.fields['primary_ip6'].widget.attrs['readonly'] = True
  308. # Limit rack choices
  309. if self.is_bound and self.data.get('site'):
  310. self.fields['rack'].queryset = Rack.objects.filter(site__pk=self.data['site'])
  311. elif self.initial.get('site'):
  312. self.fields['rack'].queryset = Rack.objects.filter(site=self.initial['site'])
  313. else:
  314. self.fields['rack'].choices = []
  315. # Rack position
  316. pk = self.instance.pk if self.instance.pk else None
  317. try:
  318. if self.is_bound and self.data.get('rack') and str(self.data.get('face')):
  319. position_choices = Rack.objects.get(pk=self.data['rack'])\
  320. .get_rack_units(face=self.data.get('face'), exclude=pk)
  321. elif self.initial.get('rack') and str(self.initial.get('face')):
  322. position_choices = Rack.objects.get(pk=self.initial['rack'])\
  323. .get_rack_units(face=self.initial.get('face'), exclude=pk)
  324. else:
  325. position_choices = []
  326. except Rack.DoesNotExist:
  327. position_choices = []
  328. self.fields['position'].choices = [('', '---------')] + [
  329. (p['id'], {
  330. 'label': p['name'],
  331. 'disabled': bool(p['device'] and p['id'] != self.initial.get('position')),
  332. }) for p in position_choices
  333. ]
  334. # Limit device_type choices
  335. if self.is_bound:
  336. self.fields['device_type'].queryset = DeviceType.objects.filter(manufacturer__pk=self.data['manufacturer'])\
  337. .select_related('manufacturer')
  338. elif self.initial.get('manufacturer'):
  339. self.fields['device_type'].queryset = DeviceType.objects.filter(manufacturer=self.initial['manufacturer'])\
  340. .select_related('manufacturer')
  341. else:
  342. self.fields['device_type'].choices = []
  343. # Disable rack assignment if this is a child device installed in a parent device
  344. if pk and self.instance.device_type.is_child_device and hasattr(self.instance, 'parent_bay'):
  345. self.fields['site'].disabled = True
  346. self.fields['rack'].disabled = True
  347. self.initial['site'] = self.instance.parent_bay.device.rack.site_id
  348. self.initial['rack'] = self.instance.parent_bay.device.rack_id
  349. class BaseDeviceFromCSVForm(forms.ModelForm):
  350. device_role = forms.ModelChoiceField(queryset=DeviceRole.objects.all(), to_field_name='name',
  351. error_messages={'invalid_choice': 'Invalid device role.'})
  352. tenant = forms.ModelChoiceField(Tenant.objects.all(), to_field_name='name', required=False,
  353. error_messages={'invalid_choice': 'Tenant not found.'})
  354. manufacturer = forms.ModelChoiceField(queryset=Manufacturer.objects.all(), to_field_name='name',
  355. error_messages={'invalid_choice': 'Invalid manufacturer.'})
  356. model_name = forms.CharField()
  357. platform = forms.ModelChoiceField(queryset=Platform.objects.all(), required=False, to_field_name='name',
  358. error_messages={'invalid_choice': 'Invalid platform.'})
  359. class Meta:
  360. fields = []
  361. model = Device
  362. def clean(self):
  363. manufacturer = self.cleaned_data.get('manufacturer')
  364. model_name = self.cleaned_data.get('model_name')
  365. # Validate device type
  366. if manufacturer and model_name:
  367. try:
  368. self.instance.device_type = DeviceType.objects.get(manufacturer=manufacturer, model=model_name)
  369. except DeviceType.DoesNotExist:
  370. self.add_error('model_name', "Invalid device type ({} {})".format(manufacturer, model_name))
  371. class DeviceFromCSVForm(BaseDeviceFromCSVForm):
  372. site = forms.ModelChoiceField(queryset=Site.objects.all(), to_field_name='name', error_messages={
  373. 'invalid_choice': 'Invalid site name.',
  374. })
  375. rack_name = forms.CharField()
  376. face = forms.CharField(required=False)
  377. class Meta(BaseDeviceFromCSVForm.Meta):
  378. fields = ['name', 'device_role', 'tenant', 'manufacturer', 'model_name', 'platform', 'serial', 'asset_tag',
  379. 'site', 'rack_name', 'position', 'face']
  380. def clean(self):
  381. super(DeviceFromCSVForm, self).clean()
  382. site = self.cleaned_data.get('site')
  383. rack_name = self.cleaned_data.get('rack_name')
  384. # Validate rack
  385. if site and rack_name:
  386. try:
  387. self.instance.rack = Rack.objects.get(site=site, name=rack_name)
  388. except Rack.DoesNotExist:
  389. self.add_error('rack_name', "Invalid rack ({})".format(rack_name))
  390. def clean_face(self):
  391. face = self.cleaned_data['face']
  392. if not face:
  393. return None
  394. try:
  395. return {
  396. 'front': 0,
  397. 'rear': 1,
  398. }[face.lower()]
  399. except KeyError:
  400. raise forms.ValidationError('Invalid rack face ({}); must be "front" or "rear".'.format(face))
  401. class ChildDeviceFromCSVForm(BaseDeviceFromCSVForm):
  402. parent = FlexibleModelChoiceField(queryset=Device.objects.all(), to_field_name='name', required=False,
  403. error_messages={'invalid_choice': 'Parent device not found.'})
  404. device_bay_name = forms.CharField(required=False)
  405. class Meta(BaseDeviceFromCSVForm.Meta):
  406. fields = ['name', 'device_role', 'tenant', 'manufacturer', 'model_name', 'platform', 'serial', 'asset_tag',
  407. 'parent', 'device_bay_name']
  408. def clean(self):
  409. super(ChildDeviceFromCSVForm, self).clean()
  410. parent = self.cleaned_data.get('parent')
  411. device_bay_name = self.cleaned_data.get('device_bay_name')
  412. # Validate device bay
  413. if parent and device_bay_name:
  414. try:
  415. device_bay = DeviceBay.objects.get(device=parent, name=device_bay_name)
  416. if device_bay.installed_device:
  417. self.add_error('device_bay_name',
  418. "Device bay ({} {}) is already occupied".format(parent, device_bay_name))
  419. else:
  420. self.instance.parent_bay = device_bay
  421. except DeviceBay.DoesNotExist:
  422. self.add_error('device_bay_name', "Parent device/bay ({} {}) not found".format(parent, device_bay_name))
  423. class DeviceImportForm(BulkImportForm, BootstrapMixin):
  424. csv = CSVDataField(csv_form=DeviceFromCSVForm)
  425. class ChildDeviceImportForm(BulkImportForm, BootstrapMixin):
  426. csv = CSVDataField(csv_form=ChildDeviceFromCSVForm)
  427. class DeviceBulkEditForm(BootstrapMixin, CustomFieldBulkEditForm):
  428. pk = forms.ModelMultipleChoiceField(queryset=Device.objects.all(), widget=forms.MultipleHiddenInput)
  429. device_type = forms.ModelChoiceField(queryset=DeviceType.objects.all(), required=False, label='Type')
  430. device_role = forms.ModelChoiceField(queryset=DeviceRole.objects.all(), required=False, label='Role')
  431. tenant = forms.ModelChoiceField(queryset=Tenant.objects.all(), required=False)
  432. platform = forms.ModelChoiceField(queryset=Platform.objects.all(), required=False)
  433. status = forms.ChoiceField(choices=FORM_STATUS_CHOICES, required=False, initial='', label='Status')
  434. serial = forms.CharField(max_length=50, required=False, label='Serial Number')
  435. class Meta:
  436. nullable_fields = ['tenant', 'platform']
  437. class DeviceFilterForm(BootstrapMixin, CustomFieldFilterForm):
  438. model = Device
  439. site = FilterChoiceField(queryset=Site.objects.annotate(filter_count=Count('racks__devices')), to_field_name='slug')
  440. rack_group_id = FilterChoiceField(queryset=RackGroup.objects.annotate(filter_count=Count('racks__devices')),
  441. label='Rack Group')
  442. role = FilterChoiceField(queryset=DeviceRole.objects.annotate(filter_count=Count('devices')), to_field_name='slug')
  443. tenant = FilterChoiceField(queryset=Tenant.objects.annotate(filter_count=Count('devices')), to_field_name='slug',
  444. null_option=(0, 'None'))
  445. device_type_id = FilterChoiceField(queryset=DeviceType.objects.select_related('manufacturer')
  446. .annotate(filter_count=Count('instances')), label='Type')
  447. platform = FilterChoiceField(queryset=Platform.objects.annotate(filter_count=Count('devices')),
  448. to_field_name='slug', null_option=(0, 'None'))
  449. status = forms.NullBooleanField(required=False, widget=forms.Select(choices=FORM_STATUS_CHOICES))
  450. #
  451. # Console ports
  452. #
  453. class ConsolePortForm(forms.ModelForm, BootstrapMixin):
  454. class Meta:
  455. model = ConsolePort
  456. fields = ['device', 'name']
  457. widgets = {
  458. 'device': forms.HiddenInput(),
  459. }
  460. class ConsolePortCreateForm(forms.Form, BootstrapMixin):
  461. name_pattern = ExpandableNameField(label='Name')
  462. class ConsoleConnectionCSVForm(forms.Form):
  463. console_server = FlexibleModelChoiceField(queryset=Device.objects.filter(device_type__is_console_server=True),
  464. to_field_name='name',
  465. error_messages={'invalid_choice': 'Console server not found'})
  466. cs_port = forms.CharField()
  467. device = FlexibleModelChoiceField(queryset=Device.objects.all(), to_field_name='name',
  468. error_messages={'invalid_choice': 'Device not found'})
  469. console_port = forms.CharField()
  470. status = forms.CharField(validators=[validate_connection_status])
  471. def clean(self):
  472. # Validate console server port
  473. if self.cleaned_data.get('console_server'):
  474. try:
  475. cs_port = ConsoleServerPort.objects.get(device=self.cleaned_data['console_server'],
  476. name=self.cleaned_data['cs_port'])
  477. if ConsolePort.objects.filter(cs_port=cs_port):
  478. raise forms.ValidationError("Console server port is already occupied (by {} {})"
  479. .format(cs_port.connected_console.device, cs_port.connected_console))
  480. except ConsoleServerPort.DoesNotExist:
  481. raise forms.ValidationError("Invalid console server port ({} {})"
  482. .format(self.cleaned_data['console_server'], self.cleaned_data['cs_port']))
  483. # Validate console port
  484. if self.cleaned_data.get('device'):
  485. try:
  486. console_port = ConsolePort.objects.get(device=self.cleaned_data['device'],
  487. name=self.cleaned_data['console_port'])
  488. if console_port.cs_port:
  489. raise forms.ValidationError("Console port is already connected (to {} {})"
  490. .format(console_port.cs_port.device, console_port.cs_port))
  491. except ConsolePort.DoesNotExist:
  492. raise forms.ValidationError("Invalid console port ({} {})"
  493. .format(self.cleaned_data['device'], self.cleaned_data['console_port']))
  494. class ConsoleConnectionImportForm(BulkImportForm, BootstrapMixin):
  495. csv = CSVDataField(csv_form=ConsoleConnectionCSVForm)
  496. def clean(self):
  497. records = self.cleaned_data.get('csv')
  498. if not records:
  499. return
  500. connection_list = []
  501. for i, record in enumerate(records, start=1):
  502. form = self.fields['csv'].csv_form(data=record)
  503. if form.is_valid():
  504. console_port = ConsolePort.objects.get(device=form.cleaned_data['device'],
  505. name=form.cleaned_data['console_port'])
  506. console_port.cs_port = ConsoleServerPort.objects.get(device=form.cleaned_data['console_server'],
  507. name=form.cleaned_data['cs_port'])
  508. if form.cleaned_data['status'] == 'planned':
  509. console_port.connection_status = CONNECTION_STATUS_PLANNED
  510. else:
  511. console_port.connection_status = CONNECTION_STATUS_CONNECTED
  512. connection_list.append(console_port)
  513. else:
  514. for field, errors in form.errors.items():
  515. for e in errors:
  516. self.add_error('csv', "Record {} {}: {}".format(i, field, e))
  517. self.cleaned_data['csv'] = connection_list
  518. class ConsolePortConnectionForm(forms.ModelForm, BootstrapMixin):
  519. rack = forms.ModelChoiceField(queryset=Rack.objects.all(), label='Rack', required=False,
  520. widget=forms.Select(attrs={'filter-for': 'console_server'}))
  521. console_server = forms.ModelChoiceField(queryset=Device.objects.all(), label='Console Server', required=False,
  522. widget=APISelect(api_url='/api/dcim/devices/?rack_id={{rack}}&is_console_server=True',
  523. attrs={'filter-for': 'cs_port'}))
  524. livesearch = forms.CharField(required=False, label='Console Server', widget=Livesearch(
  525. query_key='q', query_url='dcim-api:device_list', field_to_update='console_server')
  526. )
  527. cs_port = forms.ModelChoiceField(queryset=ConsoleServerPort.objects.all(), label='Port',
  528. widget=APISelect(api_url='/api/dcim/devices/{{console_server}}/console-server-ports/',
  529. disabled_indicator='connected_console'))
  530. class Meta:
  531. model = ConsolePort
  532. fields = ['rack', 'console_server', 'livesearch', 'cs_port', 'connection_status']
  533. labels = {
  534. 'cs_port': 'Port',
  535. 'connection_status': 'Status',
  536. }
  537. def __init__(self, *args, **kwargs):
  538. super(ConsolePortConnectionForm, self).__init__(*args, **kwargs)
  539. if not self.instance.pk:
  540. raise RuntimeError("ConsolePortConnectionForm must be initialized with an existing ConsolePort instance.")
  541. self.fields['rack'].queryset = Rack.objects.filter(site=self.instance.device.rack.site)
  542. self.fields['cs_port'].required = True
  543. self.fields['connection_status'].choices = CONNECTION_STATUS_CHOICES
  544. # Initialize console server choices
  545. if self.is_bound and self.data.get('rack'):
  546. self.fields['console_server'].queryset = Device.objects.filter(rack=self.data['rack'], device_type__is_console_server=True)
  547. elif self.initial.get('rack'):
  548. self.fields['console_server'].queryset = Device.objects.filter(rack=self.initial['rack'], device_type__is_console_server=True)
  549. else:
  550. self.fields['console_server'].choices = []
  551. # Initialize CS port choices
  552. if self.is_bound:
  553. self.fields['cs_port'].queryset = ConsoleServerPort.objects.filter(device__pk=self.data['console_server'])
  554. elif self.initial.get('console_server', None):
  555. self.fields['cs_port'].queryset = ConsoleServerPort.objects.filter(device__pk=self.initial['console_server'])
  556. else:
  557. self.fields['cs_port'].choices = []
  558. #
  559. # Console server ports
  560. #
  561. class ConsoleServerPortForm(forms.ModelForm, BootstrapMixin):
  562. class Meta:
  563. model = ConsoleServerPort
  564. fields = ['device', 'name']
  565. widgets = {
  566. 'device': forms.HiddenInput(),
  567. }
  568. class ConsoleServerPortCreateForm(forms.Form, BootstrapMixin):
  569. name_pattern = ExpandableNameField(label='Name')
  570. class ConsoleServerPortConnectionForm(forms.Form, BootstrapMixin):
  571. rack = forms.ModelChoiceField(queryset=Rack.objects.all(), label='Rack', required=False,
  572. widget=forms.Select(attrs={'filter-for': 'device'}))
  573. device = forms.ModelChoiceField(queryset=Device.objects.all(), label='Device', required=False,
  574. widget=APISelect(api_url='/api/dcim/devices/?rack_id={{rack}}',
  575. attrs={'filter-for': 'port'}))
  576. livesearch = forms.CharField(required=False, label='Device', widget=Livesearch(
  577. query_key='q', query_url='dcim-api:device_list', field_to_update='device')
  578. )
  579. port = forms.ModelChoiceField(queryset=ConsolePort.objects.all(), label='Port',
  580. widget=APISelect(api_url='/api/dcim/devices/{{device}}/console-ports/',
  581. disabled_indicator='cs_port'))
  582. connection_status = forms.BooleanField(required=False, initial=CONNECTION_STATUS_CONNECTED, label='Status',
  583. widget=forms.Select(choices=CONNECTION_STATUS_CHOICES))
  584. class Meta:
  585. fields = ['rack', 'device', 'livesearch', 'port', 'connection_status']
  586. labels = {
  587. 'connection_status': 'Status',
  588. }
  589. def __init__(self, consoleserverport, *args, **kwargs):
  590. super(ConsoleServerPortConnectionForm, self).__init__(*args, **kwargs)
  591. self.fields['rack'].queryset = Rack.objects.filter(site=consoleserverport.device.rack.site)
  592. # Initialize device choices
  593. if self.is_bound and self.data.get('rack'):
  594. self.fields['device'].queryset = Device.objects.filter(rack=self.data['rack'])
  595. elif self.initial.get('rack', None):
  596. self.fields['device'].queryset = Device.objects.filter(rack=self.initial['rack'])
  597. else:
  598. self.fields['device'].choices = []
  599. # Initialize port choices
  600. if self.is_bound:
  601. self.fields['port'].queryset = ConsolePort.objects.filter(device__pk=self.data['device'])
  602. elif self.initial.get('device', None):
  603. self.fields['port'].queryset = ConsolePort.objects.filter(device_pk=self.initial['device'])
  604. else:
  605. self.fields['port'].choices = []
  606. #
  607. # Power ports
  608. #
  609. class PowerPortForm(forms.ModelForm, BootstrapMixin):
  610. class Meta:
  611. model = PowerPort
  612. fields = ['device', 'name']
  613. widgets = {
  614. 'device': forms.HiddenInput(),
  615. }
  616. class PowerPortCreateForm(forms.Form, BootstrapMixin):
  617. name_pattern = ExpandableNameField(label='Name')
  618. class PowerConnectionCSVForm(forms.Form):
  619. pdu = FlexibleModelChoiceField(queryset=Device.objects.filter(device_type__is_pdu=True), to_field_name='name',
  620. error_messages={'invalid_choice': 'PDU not found.'})
  621. power_outlet = forms.CharField()
  622. device = FlexibleModelChoiceField(queryset=Device.objects.all(), to_field_name='name',
  623. error_messages={'invalid_choice': 'Device not found'})
  624. power_port = forms.CharField()
  625. status = forms.CharField(validators=[validate_connection_status])
  626. def clean(self):
  627. # Validate power outlet
  628. if self.cleaned_data.get('pdu'):
  629. try:
  630. power_outlet = PowerOutlet.objects.get(device=self.cleaned_data['pdu'],
  631. name=self.cleaned_data['power_outlet'])
  632. if PowerPort.objects.filter(power_outlet=power_outlet):
  633. raise forms.ValidationError("Power outlet is already occupied (by {} {})"
  634. .format(power_outlet.connected_port.device,
  635. power_outlet.connected_port))
  636. except PowerOutlet.DoesNotExist:
  637. raise forms.ValidationError("Invalid PDU port ({} {})"
  638. .format(self.cleaned_data['pdu'], self.cleaned_data['power_outlet']))
  639. # Validate power port
  640. if self.cleaned_data.get('device'):
  641. try:
  642. power_port = PowerPort.objects.get(device=self.cleaned_data['device'],
  643. name=self.cleaned_data['power_port'])
  644. if power_port.power_outlet:
  645. raise forms.ValidationError("Power port is already connected (to {} {})"
  646. .format(power_port.power_outlet.device, power_port.power_outlet))
  647. except PowerPort.DoesNotExist:
  648. raise forms.ValidationError("Invalid power port ({} {})"
  649. .format(self.cleaned_data['device'], self.cleaned_data['power_port']))
  650. class PowerConnectionImportForm(BulkImportForm, BootstrapMixin):
  651. csv = CSVDataField(csv_form=PowerConnectionCSVForm)
  652. def clean(self):
  653. records = self.cleaned_data.get('csv')
  654. if not records:
  655. return
  656. connection_list = []
  657. for i, record in enumerate(records, start=1):
  658. form = self.fields['csv'].csv_form(data=record)
  659. if form.is_valid():
  660. power_port = PowerPort.objects.get(device=form.cleaned_data['device'],
  661. name=form.cleaned_data['power_port'])
  662. power_port.power_outlet = PowerOutlet.objects.get(device=form.cleaned_data['pdu'],
  663. name=form.cleaned_data['power_outlet'])
  664. if form.cleaned_data['status'] == 'planned':
  665. power_port.connection_status = CONNECTION_STATUS_PLANNED
  666. else:
  667. power_port.connection_status = CONNECTION_STATUS_CONNECTED
  668. connection_list.append(power_port)
  669. else:
  670. for field, errors in form.errors.items():
  671. for e in errors:
  672. self.add_error('csv', "Record {} {}: {}".format(i, field, e))
  673. self.cleaned_data['csv'] = connection_list
  674. class PowerPortConnectionForm(forms.ModelForm, BootstrapMixin):
  675. rack = forms.ModelChoiceField(queryset=Rack.objects.all(), label='Rack', required=False,
  676. widget=forms.Select(attrs={'filter-for': 'pdu'}))
  677. pdu = forms.ModelChoiceField(queryset=Device.objects.all(), label='PDU', required=False,
  678. widget=APISelect(api_url='/api/dcim/devices/?rack_id={{rack}}&is_pdu=True',
  679. attrs={'filter-for': 'power_outlet'}))
  680. livesearch = forms.CharField(required=False, label='PDU', widget=Livesearch(
  681. query_key='q', query_url='dcim-api:device_list', field_to_update='pdu')
  682. )
  683. power_outlet = forms.ModelChoiceField(queryset=PowerOutlet.objects.all(), label='Outlet',
  684. widget=APISelect(api_url='/api/dcim/devices/{{pdu}}/power-outlets/',
  685. disabled_indicator='connected_port'))
  686. class Meta:
  687. model = PowerPort
  688. fields = ['rack', 'pdu', 'livesearch', 'power_outlet', 'connection_status']
  689. labels = {
  690. 'power_outlet': 'Outlet',
  691. 'connection_status': 'Status',
  692. }
  693. def __init__(self, *args, **kwargs):
  694. super(PowerPortConnectionForm, self).__init__(*args, **kwargs)
  695. if not self.instance.pk:
  696. raise RuntimeError("PowerPortConnectionForm must be initialized with an existing PowerPort instance.")
  697. self.fields['rack'].queryset = Rack.objects.filter(site=self.instance.device.rack.site)
  698. self.fields['power_outlet'].required = True
  699. self.fields['connection_status'].choices = CONNECTION_STATUS_CHOICES
  700. # Initialize PDU choices
  701. if self.is_bound and self.data.get('rack'):
  702. self.fields['pdu'].queryset = Device.objects.filter(rack=self.data['rack'], device_type__is_pdu=True)
  703. elif self.initial.get('rack', None):
  704. self.fields['pdu'].queryset = Device.objects.filter(rack=self.initial['rack'], device_type__is_pdu=True)
  705. else:
  706. self.fields['pdu'].choices = []
  707. # Initialize power outlet choices
  708. if self.is_bound:
  709. self.fields['power_outlet'].queryset = PowerOutlet.objects.filter(device__pk=self.data['pdu'])
  710. elif self.initial.get('pdu', None):
  711. self.fields['power_outlet'].queryset = PowerOutlet.objects.filter(device__pk=self.initial['pdu'])
  712. else:
  713. self.fields['power_outlet'].choices = []
  714. #
  715. # Power outlets
  716. #
  717. class PowerOutletForm(forms.ModelForm, BootstrapMixin):
  718. class Meta:
  719. model = PowerOutlet
  720. fields = ['device', 'name']
  721. widgets = {
  722. 'device': forms.HiddenInput(),
  723. }
  724. class PowerOutletCreateForm(forms.Form, BootstrapMixin):
  725. name_pattern = ExpandableNameField(label='Name')
  726. class PowerOutletConnectionForm(forms.Form, BootstrapMixin):
  727. rack = forms.ModelChoiceField(queryset=Rack.objects.all(), label='Rack', required=False,
  728. widget=forms.Select(attrs={'filter-for': 'device'}))
  729. device = forms.ModelChoiceField(queryset=Device.objects.all(), label='Device', required=False,
  730. widget=APISelect(api_url='/api/dcim/devices/?rack_id={{rack}}',
  731. attrs={'filter-for': 'port'}))
  732. livesearch = forms.CharField(required=False, label='Device', widget=Livesearch(
  733. query_key='q', query_url='dcim-api:device_list', field_to_update='device')
  734. )
  735. port = forms.ModelChoiceField(queryset=PowerPort.objects.all(), label='Port',
  736. widget=APISelect(api_url='/api/dcim/devices/{{device}}/power-ports/',
  737. disabled_indicator='power_outlet'))
  738. connection_status = forms.BooleanField(required=False, initial=CONNECTION_STATUS_CONNECTED, label='Status',
  739. widget=forms.Select(choices=CONNECTION_STATUS_CHOICES))
  740. class Meta:
  741. fields = ['rack', 'device', 'livesearch', 'port', 'connection_status']
  742. labels = {
  743. 'connection_status': 'Status',
  744. }
  745. def __init__(self, poweroutlet, *args, **kwargs):
  746. super(PowerOutletConnectionForm, self).__init__(*args, **kwargs)
  747. self.fields['rack'].queryset = Rack.objects.filter(site=poweroutlet.device.rack.site)
  748. # Initialize device choices
  749. if self.is_bound and self.data.get('rack'):
  750. self.fields['device'].queryset = Device.objects.filter(rack=self.data['rack'])
  751. elif self.initial.get('rack', None):
  752. self.fields['device'].queryset = Device.objects.filter(rack=self.initial['rack'])
  753. else:
  754. self.fields['device'].choices = []
  755. # Initialize port choices
  756. if self.is_bound:
  757. self.fields['port'].queryset = PowerPort.objects.filter(device__pk=self.data['device'])
  758. elif self.initial.get('device', None):
  759. self.fields['port'].queryset = PowerPort.objects.filter(device_pk=self.initial['device'])
  760. else:
  761. self.fields['port'].choices = []
  762. #
  763. # Interfaces
  764. #
  765. class InterfaceForm(forms.ModelForm, BootstrapMixin):
  766. class Meta:
  767. model = Interface
  768. fields = ['device', 'name', 'form_factor', 'mac_address', 'mgmt_only', 'description']
  769. widgets = {
  770. 'device': forms.HiddenInput(),
  771. }
  772. class InterfaceCreateForm(forms.ModelForm, BootstrapMixin):
  773. name_pattern = ExpandableNameField(label='Name')
  774. class Meta:
  775. model = Interface
  776. fields = ['name_pattern', 'form_factor', 'mac_address', 'mgmt_only', 'description']
  777. class InterfaceBulkCreateForm(InterfaceCreateForm, BootstrapMixin):
  778. pk = forms.ModelMultipleChoiceField(queryset=Device.objects.all(), widget=forms.MultipleHiddenInput)
  779. #
  780. # Interface connections
  781. #
  782. class InterfaceConnectionForm(forms.ModelForm, BootstrapMixin):
  783. interface_a = forms.ChoiceField(choices=[], widget=SelectWithDisabled, label='Interface')
  784. rack_b = forms.ModelChoiceField(queryset=Rack.objects.all(), label='Rack', required=False,
  785. widget=forms.Select(attrs={'filter-for': 'device_b'}))
  786. device_b = forms.ModelChoiceField(queryset=Device.objects.all(), label='Device', required=False,
  787. widget=APISelect(api_url='/api/dcim/devices/?rack_id={{rack_b}}',
  788. attrs={'filter-for': 'interface_b'}))
  789. livesearch = forms.CharField(required=False, label='Device', widget=Livesearch(
  790. query_key='q', query_url='dcim-api:device_list', field_to_update='device_b')
  791. )
  792. interface_b = forms.ModelChoiceField(queryset=Interface.objects.all(), label='Interface',
  793. widget=APISelect(api_url='/api/dcim/devices/{{device_b}}/interfaces/?type=physical',
  794. disabled_indicator='is_connected'))
  795. class Meta:
  796. model = InterfaceConnection
  797. fields = ['interface_a', 'rack_b', 'device_b', 'interface_b', 'livesearch', 'connection_status']
  798. def __init__(self, device_a, *args, **kwargs):
  799. super(InterfaceConnectionForm, self).__init__(*args, **kwargs)
  800. self.fields['rack_b'].queryset = Rack.objects.filter(site=device_a.rack.site)
  801. # Initialize interface A choices
  802. device_a_interfaces = Interface.objects.filter(device=device_a).exclude(form_factor=IFACE_FF_VIRTUAL) \
  803. .select_related('circuit', 'connected_as_a', 'connected_as_b')
  804. self.fields['interface_a'].choices = [
  805. (iface.id, {'label': iface.name, 'disabled': iface.is_connected}) for iface in device_a_interfaces
  806. ]
  807. # Initialize device_b choices if rack_b is set
  808. if self.is_bound and self.data.get('rack_b'):
  809. self.fields['device_b'].queryset = Device.objects.filter(rack__pk=self.data['rack_b'])
  810. elif self.initial.get('rack_b'):
  811. self.fields['device_b'].queryset = Device.objects.filter(rack=self.initial['rack_b'])
  812. else:
  813. self.fields['device_b'].choices = []
  814. # Initialize interface_b choices if device_b is set
  815. if self.is_bound:
  816. device_b_interfaces = Interface.objects.filter(device=self.data['device_b']) \
  817. .exclude(form_factor=IFACE_FF_VIRTUAL).select_related('circuit', 'connected_as_a', 'connected_as_b')
  818. elif self.initial.get('device_b'):
  819. device_b_interfaces = Interface.objects.filter(device=self.initial['device_b']) \
  820. .exclude(form_factor=IFACE_FF_VIRTUAL).select_related('circuit', 'connected_as_a', 'connected_as_b')
  821. else:
  822. device_b_interfaces = []
  823. self.fields['interface_b'].choices = [
  824. (iface.id, {'label': iface.name, 'disabled': iface.is_connected}) for iface in device_b_interfaces
  825. ]
  826. class InterfaceConnectionCSVForm(forms.Form):
  827. device_a = FlexibleModelChoiceField(queryset=Device.objects.all(), to_field_name='name',
  828. error_messages={'invalid_choice': 'Device A not found.'})
  829. interface_a = forms.CharField()
  830. device_b = FlexibleModelChoiceField(queryset=Device.objects.all(), to_field_name='name',
  831. error_messages={'invalid_choice': 'Device B not found.'})
  832. interface_b = forms.CharField()
  833. status = forms.CharField(validators=[validate_connection_status])
  834. def clean(self):
  835. # Validate interface A
  836. if self.cleaned_data.get('device_a'):
  837. try:
  838. interface_a = Interface.objects.get(device=self.cleaned_data['device_a'],
  839. name=self.cleaned_data['interface_a'])
  840. except Interface.DoesNotExist:
  841. raise forms.ValidationError("Invalid interface ({} {})"
  842. .format(self.cleaned_data['device_a'], self.cleaned_data['interface_a']))
  843. try:
  844. InterfaceConnection.objects.get(Q(interface_a=interface_a) | Q(interface_b=interface_a))
  845. raise forms.ValidationError("{} {} is already connected"
  846. .format(self.cleaned_data['device_a'], self.cleaned_data['interface_a']))
  847. except InterfaceConnection.DoesNotExist:
  848. pass
  849. # Validate interface B
  850. if self.cleaned_data.get('device_b'):
  851. try:
  852. interface_b = Interface.objects.get(device=self.cleaned_data['device_b'],
  853. name=self.cleaned_data['interface_b'])
  854. except Interface.DoesNotExist:
  855. raise forms.ValidationError("Invalid interface ({} {})"
  856. .format(self.cleaned_data['device_b'], self.cleaned_data['interface_b']))
  857. try:
  858. InterfaceConnection.objects.get(Q(interface_a=interface_b) | Q(interface_b=interface_b))
  859. raise forms.ValidationError("{} {} is already connected"
  860. .format(self.cleaned_data['device_b'], self.cleaned_data['interface_b']))
  861. except InterfaceConnection.DoesNotExist:
  862. pass
  863. class InterfaceConnectionImportForm(BulkImportForm, BootstrapMixin):
  864. csv = CSVDataField(csv_form=InterfaceConnectionCSVForm)
  865. def clean(self):
  866. records = self.cleaned_data.get('csv')
  867. if not records:
  868. return
  869. connection_list = []
  870. occupied_interfaces = []
  871. for i, record in enumerate(records, start=1):
  872. form = self.fields['csv'].csv_form(data=record)
  873. if form.is_valid():
  874. interface_a = Interface.objects.get(device=form.cleaned_data['device_a'],
  875. name=form.cleaned_data['interface_a'])
  876. if interface_a in occupied_interfaces:
  877. raise forms.ValidationError("{} {} found in multiple connections"
  878. .format(interface_a.device.name, interface_a.name))
  879. interface_b = Interface.objects.get(device=form.cleaned_data['device_b'],
  880. name=form.cleaned_data['interface_b'])
  881. if interface_b in occupied_interfaces:
  882. raise forms.ValidationError("{} {} found in multiple connections"
  883. .format(interface_b.device.name, interface_b.name))
  884. connection = InterfaceConnection(interface_a=interface_a, interface_b=interface_b)
  885. if form.cleaned_data['status'] == 'planned':
  886. connection.connection_status = CONNECTION_STATUS_PLANNED
  887. else:
  888. connection.connection_status = CONNECTION_STATUS_CONNECTED
  889. connection_list.append(connection)
  890. occupied_interfaces.append(interface_a)
  891. occupied_interfaces.append(interface_b)
  892. else:
  893. for field, errors in form.errors.items():
  894. for e in errors:
  895. self.add_error('csv', "Record {} {}: {}".format(i, field, e))
  896. self.cleaned_data['csv'] = connection_list
  897. class InterfaceConnectionDeletionForm(forms.Form, BootstrapMixin):
  898. confirm = forms.BooleanField(required=True)
  899. # Used for HTTP redirect upon successful deletion
  900. device = forms.ModelChoiceField(queryset=Device.objects.all(), widget=forms.HiddenInput(), required=False)
  901. #
  902. # Device bays
  903. #
  904. class DeviceBayForm(forms.ModelForm, BootstrapMixin):
  905. class Meta:
  906. model = DeviceBay
  907. fields = ['device', 'name']
  908. widgets = {
  909. 'device': forms.HiddenInput(),
  910. }
  911. class DeviceBayCreateForm(forms.Form, BootstrapMixin):
  912. name_pattern = ExpandableNameField(label='Name')
  913. class PopulateDeviceBayForm(forms.Form, BootstrapMixin):
  914. installed_device = forms.ModelChoiceField(queryset=Device.objects.all(), label='Child Device',
  915. help_text="Child devices must first be created within the rack occupied "
  916. "by the parent device. Then they can be assigned to a bay.")
  917. def __init__(self, device_bay, *args, **kwargs):
  918. super(PopulateDeviceBayForm, self).__init__(*args, **kwargs)
  919. children_queryset = Device.objects.filter(rack=device_bay.device.rack,
  920. parent_bay__isnull=True,
  921. device_type__u_height=0,
  922. device_type__subdevice_role=SUBDEVICE_ROLE_CHILD)\
  923. .exclude(pk=device_bay.device.pk)
  924. self.fields['installed_device'].queryset = children_queryset
  925. #
  926. # Connections
  927. #
  928. class ConsoleConnectionFilterForm(forms.Form, BootstrapMixin):
  929. site = forms.ModelChoiceField(required=False, queryset=Site.objects.all(), to_field_name='slug')
  930. class PowerConnectionFilterForm(forms.Form, BootstrapMixin):
  931. site = forms.ModelChoiceField(required=False, queryset=Site.objects.all(), to_field_name='slug')
  932. class InterfaceConnectionFilterForm(forms.Form, BootstrapMixin):
  933. site = forms.ModelChoiceField(required=False, queryset=Site.objects.all(), to_field_name='slug')
  934. #
  935. # IP addresses
  936. #
  937. class IPAddressForm(forms.ModelForm, BootstrapMixin):
  938. set_as_primary = forms.BooleanField(label='Set as primary IP for device', required=False)
  939. class Meta:
  940. model = IPAddress
  941. fields = ['address', 'vrf', 'interface', 'set_as_primary']
  942. help_texts = {
  943. 'address': 'IPv4 or IPv6 address (with mask)'
  944. }
  945. def __init__(self, device, *args, **kwargs):
  946. super(IPAddressForm, self).__init__(*args, **kwargs)
  947. self.fields['vrf'].empty_label = 'Global'
  948. self.fields['interface'].queryset = device.interfaces.all()
  949. self.fields['interface'].required = True
  950. # If this device does not have any IP addresses assigned, default to setting the first IP as its primary
  951. if not IPAddress.objects.filter(interface__device=device).count():
  952. self.fields['set_as_primary'].initial = True
  953. #
  954. # Interfaces
  955. #
  956. class ModuleForm(forms.ModelForm, BootstrapMixin):
  957. class Meta:
  958. model = Module
  959. fields = ['name', 'manufacturer', 'part_id', 'serial']