forms.py 53 KB

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