forms.py 49 KB

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