forms.py 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631
  1. from django import forms
  2. from django.db.models import Count
  3. from dcim.models import Site, Rack, Device, Interface
  4. from extras.forms import CustomFieldForm, CustomFieldBulkEditForm, CustomFieldFilterForm
  5. from tenancy.models import Tenant
  6. from utilities.forms import (
  7. APISelect, BootstrapMixin, BulkImportForm, CSVDataField, ExpandableIPAddressField, FilterChoiceField, Livesearch,
  8. SlugField, add_blank_choice,
  9. )
  10. from .models import (
  11. Aggregate, IPAddress, IPADDRESS_STATUS_CHOICES, Prefix, PREFIX_STATUS_CHOICES, RIR, Role, Service, VLAN,
  12. VLANGroup, VLAN_STATUS_CHOICES, VRF,
  13. )
  14. IP_FAMILY_CHOICES = [
  15. ('', 'All'),
  16. (4, 'IPv4'),
  17. (6, 'IPv6'),
  18. ]
  19. #
  20. # VRFs
  21. #
  22. class VRFForm(BootstrapMixin, CustomFieldForm):
  23. class Meta:
  24. model = VRF
  25. fields = ['name', 'rd', 'tenant', 'enforce_unique', 'description']
  26. labels = {
  27. 'rd': "RD",
  28. }
  29. help_texts = {
  30. 'rd': "Route distinguisher in any format",
  31. }
  32. class VRFFromCSVForm(forms.ModelForm):
  33. tenant = forms.ModelChoiceField(Tenant.objects.all(), to_field_name='name', required=False,
  34. error_messages={'invalid_choice': 'Tenant not found.'})
  35. class Meta:
  36. model = VRF
  37. fields = ['name', 'rd', 'tenant', 'enforce_unique', 'description']
  38. class VRFImportForm(BootstrapMixin, BulkImportForm):
  39. csv = CSVDataField(csv_form=VRFFromCSVForm)
  40. class VRFBulkEditForm(BootstrapMixin, CustomFieldBulkEditForm):
  41. pk = forms.ModelMultipleChoiceField(queryset=VRF.objects.all(), widget=forms.MultipleHiddenInput)
  42. tenant = forms.ModelChoiceField(queryset=Tenant.objects.all(), required=False)
  43. description = forms.CharField(max_length=100, required=False)
  44. class Meta:
  45. nullable_fields = ['tenant', 'description']
  46. class VRFFilterForm(BootstrapMixin, CustomFieldFilterForm):
  47. model = VRF
  48. q = forms.CharField(required=False, label='Search')
  49. tenant = FilterChoiceField(queryset=Tenant.objects.annotate(filter_count=Count('vrfs')), to_field_name='slug',
  50. null_option=(0, None))
  51. #
  52. # RIRs
  53. #
  54. class RIRForm(BootstrapMixin, forms.ModelForm):
  55. slug = SlugField()
  56. class Meta:
  57. model = RIR
  58. fields = ['name', 'slug', 'is_private']
  59. class RIRFilterForm(BootstrapMixin, forms.Form):
  60. is_private = forms.NullBooleanField(required=False, label='Private', widget=forms.Select(choices=[
  61. ('', '---------'),
  62. ('True', 'Yes'),
  63. ('False', 'No'),
  64. ]))
  65. #
  66. # Aggregates
  67. #
  68. class AggregateForm(BootstrapMixin, CustomFieldForm):
  69. class Meta:
  70. model = Aggregate
  71. fields = ['prefix', 'rir', 'date_added', 'description']
  72. help_texts = {
  73. 'prefix': "IPv4 or IPv6 network",
  74. 'rir': "Regional Internet Registry responsible for this prefix",
  75. 'date_added': "Format: YYYY-MM-DD",
  76. }
  77. class AggregateFromCSVForm(forms.ModelForm):
  78. rir = forms.ModelChoiceField(queryset=RIR.objects.all(), to_field_name='name',
  79. error_messages={'invalid_choice': 'RIR not found.'})
  80. class Meta:
  81. model = Aggregate
  82. fields = ['prefix', 'rir', 'date_added', 'description']
  83. class AggregateImportForm(BootstrapMixin, BulkImportForm):
  84. csv = CSVDataField(csv_form=AggregateFromCSVForm)
  85. class AggregateBulkEditForm(BootstrapMixin, CustomFieldBulkEditForm):
  86. pk = forms.ModelMultipleChoiceField(queryset=Aggregate.objects.all(), widget=forms.MultipleHiddenInput)
  87. rir = forms.ModelChoiceField(queryset=RIR.objects.all(), required=False, label='RIR')
  88. date_added = forms.DateField(required=False)
  89. description = forms.CharField(max_length=100, required=False)
  90. class Meta:
  91. nullable_fields = ['date_added', 'description']
  92. class AggregateFilterForm(BootstrapMixin, CustomFieldFilterForm):
  93. model = Aggregate
  94. q = forms.CharField(required=False, label='Search')
  95. family = forms.ChoiceField(required=False, choices=IP_FAMILY_CHOICES, label='Address Family')
  96. rir = FilterChoiceField(queryset=RIR.objects.annotate(filter_count=Count('aggregates')), to_field_name='slug',
  97. label='RIR')
  98. #
  99. # Roles
  100. #
  101. class RoleForm(BootstrapMixin, forms.ModelForm):
  102. slug = SlugField()
  103. class Meta:
  104. model = Role
  105. fields = ['name', 'slug']
  106. #
  107. # Prefixes
  108. #
  109. class PrefixForm(BootstrapMixin, CustomFieldForm):
  110. site = forms.ModelChoiceField(queryset=Site.objects.all(), required=False, label='Site',
  111. widget=forms.Select(attrs={'filter-for': 'vlan'}))
  112. vlan = forms.ModelChoiceField(queryset=VLAN.objects.all(), required=False, label='VLAN',
  113. widget=APISelect(api_url='/api/ipam/vlans/?site_id={{site}}',
  114. display_field='display_name'))
  115. class Meta:
  116. model = Prefix
  117. fields = ['prefix', 'vrf', 'tenant', 'site', 'vlan', 'status', 'role', 'is_pool', 'description']
  118. def __init__(self, *args, **kwargs):
  119. super(PrefixForm, self).__init__(*args, **kwargs)
  120. self.fields['vrf'].empty_label = 'Global'
  121. # Initialize field without choices to avoid pulling all VLANs from the database
  122. if self.is_bound and self.data.get('site'):
  123. self.fields['vlan'].queryset = VLAN.objects.filter(site__pk=self.data['site'])
  124. elif self.initial.get('site'):
  125. self.fields['vlan'].queryset = VLAN.objects.filter(site=self.initial['site'])
  126. else:
  127. self.fields['vlan'].choices = []
  128. class PrefixFromCSVForm(forms.ModelForm):
  129. vrf = forms.ModelChoiceField(queryset=VRF.objects.all(), required=False, to_field_name='rd',
  130. error_messages={'invalid_choice': 'VRF not found.'})
  131. tenant = forms.ModelChoiceField(Tenant.objects.all(), to_field_name='name', required=False,
  132. error_messages={'invalid_choice': 'Tenant not found.'})
  133. site = forms.ModelChoiceField(queryset=Site.objects.all(), required=False, to_field_name='name',
  134. error_messages={'invalid_choice': 'Site not found.'})
  135. vlan_group_name = forms.CharField(required=False)
  136. vlan_vid = forms.IntegerField(required=False)
  137. status_name = forms.ChoiceField(choices=[(s[1], s[0]) for s in PREFIX_STATUS_CHOICES])
  138. role = forms.ModelChoiceField(queryset=Role.objects.all(), required=False, to_field_name='name',
  139. error_messages={'invalid_choice': 'Invalid role.'})
  140. class Meta:
  141. model = Prefix
  142. fields = ['prefix', 'vrf', 'tenant', 'site', 'vlan_group_name', 'vlan_vid', 'status_name', 'role', 'is_pool',
  143. 'description']
  144. def clean(self):
  145. super(PrefixFromCSVForm, self).clean()
  146. site = self.cleaned_data.get('site')
  147. vlan_group_name = self.cleaned_data.get('vlan_group_name')
  148. vlan_vid = self.cleaned_data.get('vlan_vid')
  149. # Validate VLAN
  150. vlan_group = None
  151. if vlan_group_name:
  152. try:
  153. vlan_group = VLANGroup.objects.get(site=site, name=vlan_group_name)
  154. except VLANGroup.DoesNotExist:
  155. self.add_error('vlan_group_name', "Invalid VLAN group ({} - {}).".format(site, vlan_group_name))
  156. if vlan_vid and vlan_group:
  157. try:
  158. self.instance.vlan = VLAN.objects.get(group=vlan_group, vid=vlan_vid)
  159. except VLAN.DoesNotExist:
  160. self.add_error('vlan_vid', "Invalid VLAN ID ({} - {}).".format(vlan_group, vlan_vid))
  161. elif vlan_vid and site:
  162. try:
  163. self.instance.vlan = VLAN.objects.get(site=site, vid=vlan_vid)
  164. except VLAN.DoesNotExist:
  165. self.add_error('vlan_vid', "Invalid VLAN ID ({}) for site {}.".format(vlan_vid, site))
  166. except VLAN.MultipleObjectsReturned:
  167. self.add_error('vlan_vid', "Multiple VLANs found ({} - VID {})".format(site, vlan_vid))
  168. elif vlan_vid:
  169. self.add_error('vlan_vid', "Must specify site and/or VLAN group when assigning a VLAN.")
  170. def save(self, *args, **kwargs):
  171. # Assign Prefix status by name
  172. self.instance.status = dict(self.fields['status_name'].choices)[self.cleaned_data['status_name']]
  173. return super(PrefixFromCSVForm, self).save(*args, **kwargs)
  174. class PrefixImportForm(BootstrapMixin, BulkImportForm):
  175. csv = CSVDataField(csv_form=PrefixFromCSVForm)
  176. class PrefixBulkEditForm(BootstrapMixin, CustomFieldBulkEditForm):
  177. pk = forms.ModelMultipleChoiceField(queryset=Prefix.objects.all(), widget=forms.MultipleHiddenInput)
  178. site = forms.ModelChoiceField(queryset=Site.objects.all(), required=False)
  179. vrf = forms.ModelChoiceField(queryset=VRF.objects.all(), required=False, label='VRF')
  180. tenant = forms.ModelChoiceField(queryset=Tenant.objects.all(), required=False)
  181. status = forms.ChoiceField(choices=add_blank_choice(PREFIX_STATUS_CHOICES), required=False)
  182. role = forms.ModelChoiceField(queryset=Role.objects.all(), required=False)
  183. description = forms.CharField(max_length=100, required=False)
  184. class Meta:
  185. nullable_fields = ['site', 'vrf', 'tenant', 'role', 'description']
  186. def prefix_status_choices():
  187. status_counts = {}
  188. for status in Prefix.objects.values('status').annotate(count=Count('status')).order_by('status'):
  189. status_counts[status['status']] = status['count']
  190. return [(s[0], u'{} ({})'.format(s[1], status_counts.get(s[0], 0))) for s in PREFIX_STATUS_CHOICES]
  191. class PrefixFilterForm(BootstrapMixin, CustomFieldFilterForm):
  192. model = Prefix
  193. q = forms.CharField(required=False, label='Search')
  194. parent = forms.CharField(required=False, label='Parent Prefix', widget=forms.TextInput(attrs={
  195. 'placeholder': 'Prefix',
  196. }))
  197. family = forms.ChoiceField(required=False, choices=IP_FAMILY_CHOICES, label='Address Family')
  198. vrf = FilterChoiceField(queryset=VRF.objects.annotate(filter_count=Count('prefixes')), to_field_name='rd',
  199. label='VRF', null_option=(0, 'Global'))
  200. tenant = FilterChoiceField(queryset=Tenant.objects.annotate(filter_count=Count('prefixes')), to_field_name='slug',
  201. null_option=(0, 'None'))
  202. status = forms.MultipleChoiceField(choices=prefix_status_choices, required=False)
  203. site = FilterChoiceField(queryset=Site.objects.annotate(filter_count=Count('prefixes')), to_field_name='slug',
  204. null_option=(0, 'None'))
  205. role = FilterChoiceField(queryset=Role.objects.annotate(filter_count=Count('prefixes')), to_field_name='slug',
  206. null_option=(0, 'None'))
  207. expand = forms.BooleanField(required=False, label='Expand prefix hierarchy')
  208. #
  209. # IP addresses
  210. #
  211. class IPAddressForm(BootstrapMixin, CustomFieldForm):
  212. nat_site = forms.ModelChoiceField(queryset=Site.objects.all(), required=False, label='Site',
  213. widget=forms.Select(attrs={'filter-for': 'nat_device'}))
  214. nat_device = forms.ModelChoiceField(queryset=Device.objects.all(), required=False, label='Device',
  215. widget=APISelect(api_url='/api/dcim/devices/?site_id={{nat_site}}',
  216. display_field='display_name',
  217. attrs={'filter-for': 'nat_inside'}))
  218. livesearch = forms.CharField(required=False, label='IP Address', widget=Livesearch(
  219. query_key='q', query_url='ipam-api:ipaddress_list', field_to_update='nat_inside', obj_label='address')
  220. )
  221. class Meta:
  222. model = IPAddress
  223. fields = ['address', 'vrf', 'tenant', 'status', 'nat_inside', 'description']
  224. widgets = {
  225. 'nat_inside': APISelect(api_url='/api/ipam/ip-addresses/?device_id={{nat_device}}', display_field='address')
  226. }
  227. def __init__(self, *args, **kwargs):
  228. super(IPAddressForm, self).__init__(*args, **kwargs)
  229. self.fields['vrf'].empty_label = 'Global'
  230. if self.instance.nat_inside:
  231. nat_inside = self.instance.nat_inside
  232. # If the IP is assigned to an interface, populate site/device fields accordingly
  233. if self.instance.nat_inside.interface:
  234. self.initial['nat_site'] = self.instance.nat_inside.interface.device.site.pk
  235. self.initial['nat_device'] = self.instance.nat_inside.interface.device.pk
  236. self.fields['nat_device'].queryset = Device.objects.filter(
  237. rack__site=nat_inside.interface.device.site)
  238. self.fields['nat_inside'].queryset = IPAddress.objects.filter(
  239. interface__device=nat_inside.interface.device)
  240. else:
  241. self.fields['nat_inside'].queryset = IPAddress.objects.filter(pk=nat_inside.pk)
  242. else:
  243. # Initialize nat_device choices if nat_site is set
  244. if self.is_bound and self.data.get('nat_site'):
  245. self.fields['nat_device'].queryset = Device.objects.filter(rack__site__pk=self.data['nat_site'])
  246. elif self.initial.get('nat_site'):
  247. self.fields['nat_device'].queryset = Device.objects.filter(rack__site=self.initial['nat_site'])
  248. else:
  249. self.fields['nat_device'].choices = []
  250. # Initialize nat_inside choices if nat_device is set
  251. if self.is_bound and self.data.get('nat_device'):
  252. self.fields['nat_inside'].queryset = IPAddress.objects.filter(
  253. interface__device__pk=self.data['nat_device'])
  254. elif self.initial.get('nat_device'):
  255. self.fields['nat_inside'].queryset = IPAddress.objects.filter(
  256. interface__device__pk=self.initial['nat_device'])
  257. else:
  258. self.fields['nat_inside'].choices = []
  259. class IPAddressBulkAddForm(BootstrapMixin, forms.Form):
  260. address = ExpandableIPAddressField()
  261. vrf = forms.ModelChoiceField(queryset=VRF.objects.all(), required=False, label='VRF', empty_label='Global')
  262. tenant = forms.ModelChoiceField(queryset=Tenant.objects.all(), required=False)
  263. status = forms.ChoiceField(choices=IPADDRESS_STATUS_CHOICES)
  264. description = forms.CharField(max_length=100, required=False)
  265. class IPAddressAssignForm(BootstrapMixin, forms.Form):
  266. site = forms.ModelChoiceField(
  267. queryset=Site.objects.all(),
  268. label='Site',
  269. required=False,
  270. widget=forms.Select(
  271. attrs={'filter-for': 'rack'}
  272. )
  273. )
  274. rack = forms.ModelChoiceField(
  275. queryset=Rack.objects.all(),
  276. label='Rack',
  277. required=False,
  278. widget=APISelect(
  279. api_url='/api/dcim/racks/?site_id={{site}}',
  280. display_field='display_name',
  281. attrs={'filter-for': 'device', 'nullable': 'true'}
  282. )
  283. )
  284. device = forms.ModelChoiceField(
  285. queryset=Device.objects.all(),
  286. label='Device',
  287. required=False,
  288. widget=APISelect(
  289. api_url='/api/dcim/devices/?site_id={{site}}&rack_id={{rack}}',
  290. display_field='display_name',
  291. attrs={'filter-for': 'interface'}
  292. )
  293. )
  294. livesearch = forms.CharField(
  295. required=False,
  296. label='Device',
  297. widget=Livesearch(
  298. query_key='q',
  299. query_url='dcim-api:device_list',
  300. field_to_update='device'
  301. )
  302. )
  303. interface = forms.ModelChoiceField(
  304. queryset=Interface.objects.all(),
  305. label='Interface',
  306. widget=APISelect(
  307. api_url='/api/dcim/devices/{{device}}/interfaces/'
  308. )
  309. )
  310. set_as_primary = forms.BooleanField(
  311. label='Set as primary IP for device',
  312. required=False
  313. )
  314. def __init__(self, *args, **kwargs):
  315. super(IPAddressAssignForm, self).__init__(*args, **kwargs)
  316. self.fields['rack'].choices = []
  317. self.fields['device'].choices = []
  318. self.fields['interface'].choices = []
  319. class IPAddressFromCSVForm(forms.ModelForm):
  320. vrf = forms.ModelChoiceField(queryset=VRF.objects.all(), required=False, to_field_name='rd',
  321. error_messages={'invalid_choice': 'VRF not found.'})
  322. tenant = forms.ModelChoiceField(Tenant.objects.all(), to_field_name='name', required=False,
  323. error_messages={'invalid_choice': 'Tenant not found.'})
  324. status_name = forms.ChoiceField(choices=[(s[1], s[0]) for s in IPADDRESS_STATUS_CHOICES])
  325. device = forms.ModelChoiceField(queryset=Device.objects.all(), required=False, to_field_name='name',
  326. error_messages={'invalid_choice': 'Device not found.'})
  327. interface_name = forms.CharField(required=False)
  328. is_primary = forms.BooleanField(required=False)
  329. class Meta:
  330. model = IPAddress
  331. fields = ['address', 'vrf', 'tenant', 'status_name', 'device', 'interface_name', 'is_primary', 'description']
  332. def clean(self):
  333. device = self.cleaned_data.get('device')
  334. interface_name = self.cleaned_data.get('interface_name')
  335. is_primary = self.cleaned_data.get('is_primary')
  336. # Validate interface
  337. if device and interface_name:
  338. try:
  339. Interface.objects.get(device=device, name=interface_name)
  340. except Interface.DoesNotExist:
  341. self.add_error('interface_name', "Invalid interface ({}) for {}".format(interface_name, device))
  342. elif device and not interface_name:
  343. self.add_error('interface_name', "Device set ({}) but interface missing".format(device))
  344. elif interface_name and not device:
  345. self.add_error('device', "Interface set ({}) but device missing or invalid".format(interface_name))
  346. # Validate is_primary
  347. if is_primary and not device:
  348. self.add_error('is_primary', "No device specified; cannot set as primary IP")
  349. def save(self, *args, **kwargs):
  350. # Assign status by name
  351. self.instance.status = dict(self.fields['status_name'].choices)[self.cleaned_data['status_name']]
  352. # Set interface
  353. if self.cleaned_data['device'] and self.cleaned_data['interface_name']:
  354. self.instance.interface = Interface.objects.get(device=self.cleaned_data['device'],
  355. name=self.cleaned_data['interface_name'])
  356. # Set as primary for device
  357. if self.cleaned_data['is_primary']:
  358. if self.instance.address.version == 4:
  359. self.instance.primary_ip4_for = self.cleaned_data['device']
  360. elif self.instance.address.version == 6:
  361. self.instance.primary_ip6_for = self.cleaned_data['device']
  362. return super(IPAddressFromCSVForm, self).save(*args, **kwargs)
  363. class IPAddressImportForm(BootstrapMixin, BulkImportForm):
  364. csv = CSVDataField(csv_form=IPAddressFromCSVForm)
  365. class IPAddressBulkEditForm(BootstrapMixin, CustomFieldBulkEditForm):
  366. pk = forms.ModelMultipleChoiceField(queryset=IPAddress.objects.all(), widget=forms.MultipleHiddenInput)
  367. vrf = forms.ModelChoiceField(queryset=VRF.objects.all(), required=False, label='VRF')
  368. tenant = forms.ModelChoiceField(queryset=Tenant.objects.all(), required=False)
  369. status = forms.ChoiceField(choices=add_blank_choice(IPADDRESS_STATUS_CHOICES), required=False)
  370. description = forms.CharField(max_length=100, required=False)
  371. class Meta:
  372. nullable_fields = ['vrf', 'tenant', 'description']
  373. def ipaddress_status_choices():
  374. status_counts = {}
  375. for status in IPAddress.objects.values('status').annotate(count=Count('status')).order_by('status'):
  376. status_counts[status['status']] = status['count']
  377. return [(s[0], u'{} ({})'.format(s[1], status_counts.get(s[0], 0))) for s in IPADDRESS_STATUS_CHOICES]
  378. class IPAddressFilterForm(BootstrapMixin, CustomFieldFilterForm):
  379. model = IPAddress
  380. q = forms.CharField(required=False, label='Search')
  381. parent = forms.CharField(required=False, label='Parent Prefix', widget=forms.TextInput(attrs={
  382. 'placeholder': 'Prefix',
  383. }))
  384. family = forms.ChoiceField(required=False, choices=IP_FAMILY_CHOICES, label='Address Family')
  385. vrf = FilterChoiceField(queryset=VRF.objects.annotate(filter_count=Count('ip_addresses')), to_field_name='rd',
  386. label='VRF', null_option=(0, 'Global'))
  387. tenant = FilterChoiceField(queryset=Tenant.objects.annotate(filter_count=Count('ip_addresses')),
  388. to_field_name='slug', null_option=(0, 'None'))
  389. status = forms.MultipleChoiceField(choices=ipaddress_status_choices, required=False)
  390. #
  391. # VLAN groups
  392. #
  393. class VLANGroupForm(BootstrapMixin, forms.ModelForm):
  394. slug = SlugField()
  395. class Meta:
  396. model = VLANGroup
  397. fields = ['site', 'name', 'slug']
  398. class VLANGroupFilterForm(BootstrapMixin, forms.Form):
  399. site = FilterChoiceField(queryset=Site.objects.annotate(filter_count=Count('vlan_groups')), to_field_name='slug')
  400. #
  401. # VLANs
  402. #
  403. class VLANForm(BootstrapMixin, CustomFieldForm):
  404. group = forms.ModelChoiceField(queryset=VLANGroup.objects.all(), required=False, label='Group', widget=APISelect(
  405. api_url='/api/ipam/vlan-groups/?site_id={{site}}',
  406. ))
  407. class Meta:
  408. model = VLAN
  409. fields = ['site', 'group', 'vid', 'name', 'tenant', 'status', 'role', 'description']
  410. help_texts = {
  411. 'site': "The site at which this VLAN exists",
  412. 'group': "VLAN group (optional)",
  413. 'vid': "Configured VLAN ID",
  414. 'name': "Configured VLAN name",
  415. 'status': "Operational status of this VLAN",
  416. 'role': "The primary function of this VLAN",
  417. }
  418. widgets = {
  419. 'site': forms.Select(attrs={'filter-for': 'group'}),
  420. }
  421. def __init__(self, *args, **kwargs):
  422. super(VLANForm, self).__init__(*args, **kwargs)
  423. # Limit VLAN group choices
  424. if self.is_bound and self.data.get('site'):
  425. self.fields['group'].queryset = VLANGroup.objects.filter(site__pk=self.data['site'])
  426. elif self.initial.get('site'):
  427. self.fields['group'].queryset = VLANGroup.objects.filter(site=self.initial['site'])
  428. else:
  429. self.fields['group'].choices = []
  430. class VLANFromCSVForm(forms.ModelForm):
  431. site = forms.ModelChoiceField(queryset=Site.objects.all(), to_field_name='name',
  432. error_messages={'invalid_choice': 'Site not found.'})
  433. group = forms.ModelChoiceField(queryset=VLANGroup.objects.all(), required=False, to_field_name='name',
  434. error_messages={'invalid_choice': 'VLAN group not found.'})
  435. tenant = forms.ModelChoiceField(Tenant.objects.all(), to_field_name='name', required=False,
  436. error_messages={'invalid_choice': 'Tenant not found.'})
  437. status_name = forms.ChoiceField(choices=[(s[1], s[0]) for s in VLAN_STATUS_CHOICES])
  438. role = forms.ModelChoiceField(queryset=Role.objects.all(), required=False, to_field_name='name',
  439. error_messages={'invalid_choice': 'Invalid role.'})
  440. class Meta:
  441. model = VLAN
  442. fields = ['site', 'group', 'vid', 'name', 'tenant', 'status_name', 'role', 'description']
  443. def save(self, *args, **kwargs):
  444. m = super(VLANFromCSVForm, self).save(commit=False)
  445. # Assign VLAN status by name
  446. m.status = dict(self.fields['status_name'].choices)[self.cleaned_data['status_name']]
  447. if kwargs.get('commit'):
  448. m.save()
  449. return m
  450. class VLANImportForm(BootstrapMixin, BulkImportForm):
  451. csv = CSVDataField(csv_form=VLANFromCSVForm)
  452. class VLANBulkEditForm(BootstrapMixin, CustomFieldBulkEditForm):
  453. pk = forms.ModelMultipleChoiceField(queryset=VLAN.objects.all(), widget=forms.MultipleHiddenInput)
  454. site = forms.ModelChoiceField(queryset=Site.objects.all(), required=False)
  455. group = forms.ModelChoiceField(queryset=VLANGroup.objects.all(), required=False)
  456. tenant = forms.ModelChoiceField(queryset=Tenant.objects.all(), required=False)
  457. status = forms.ChoiceField(choices=add_blank_choice(VLAN_STATUS_CHOICES), required=False)
  458. role = forms.ModelChoiceField(queryset=Role.objects.all(), required=False)
  459. description = forms.CharField(max_length=100, required=False)
  460. class Meta:
  461. nullable_fields = ['group', 'tenant', 'role', 'description']
  462. def vlan_status_choices():
  463. status_counts = {}
  464. for status in VLAN.objects.values('status').annotate(count=Count('status')).order_by('status'):
  465. status_counts[status['status']] = status['count']
  466. return [(s[0], u'{} ({})'.format(s[1], status_counts.get(s[0], 0))) for s in VLAN_STATUS_CHOICES]
  467. class VLANFilterForm(BootstrapMixin, CustomFieldFilterForm):
  468. model = VLAN
  469. q = forms.CharField(required=False, label='Search')
  470. site = FilterChoiceField(queryset=Site.objects.annotate(filter_count=Count('vlans')), to_field_name='slug')
  471. group_id = FilterChoiceField(queryset=VLANGroup.objects.annotate(filter_count=Count('vlans')), label='VLAN group',
  472. null_option=(0, 'None'))
  473. tenant = FilterChoiceField(queryset=Tenant.objects.annotate(filter_count=Count('vlans')), to_field_name='slug',
  474. null_option=(0, 'None'))
  475. status = forms.MultipleChoiceField(choices=vlan_status_choices, required=False)
  476. role = FilterChoiceField(queryset=Role.objects.annotate(filter_count=Count('vlans')), to_field_name='slug',
  477. null_option=(0, 'None'))
  478. #
  479. # Services
  480. #
  481. class ServiceForm(BootstrapMixin, forms.ModelForm):
  482. class Meta:
  483. model = Service
  484. fields = ['name', 'protocol', 'port', 'ipaddresses', 'description']
  485. help_texts = {
  486. 'ipaddresses': "IP address assignment is optional. If no IPs are selected, the service is assumed to be "
  487. "reachable via all IPs assigned to the device.",
  488. }
  489. def __init__(self, *args, **kwargs):
  490. super(ServiceForm, self).__init__(*args, **kwargs)
  491. # Limit IP address choices to those assigned to interfaces of the parent device
  492. self.fields['ipaddresses'].queryset = IPAddress.objects.filter(interface__device=self.instance.device)