forms.py 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476
  1. from netaddr import IPNetwork
  2. from django import forms
  3. from django.db.models import Count
  4. from dcim.models import Site, Device, Interface
  5. from utilities.forms import (
  6. BootstrapMixin, ConfirmationForm, APISelect, Livesearch, CSVDataField, BulkImportForm, SlugField,
  7. )
  8. from .models import (
  9. Aggregate, IPAddress, Prefix, PREFIX_STATUS_CHOICES, RIR, Role, VLAN, VLAN_STATUS_CHOICES, VRF,
  10. )
  11. #
  12. # VRFs
  13. #
  14. class VRFForm(forms.ModelForm, BootstrapMixin):
  15. class Meta:
  16. model = VRF
  17. fields = ['name', 'rd', 'description']
  18. labels = {
  19. 'rd': "RD",
  20. }
  21. help_texts = {
  22. 'rd': "Route distinguisher in any format",
  23. }
  24. class VRFFromCSVForm(forms.ModelForm):
  25. class Meta:
  26. model = VRF
  27. fields = ['name', 'rd', 'description']
  28. class VRFImportForm(BulkImportForm, BootstrapMixin):
  29. csv = CSVDataField(csv_form=VRFFromCSVForm)
  30. class VRFBulkEditForm(forms.Form, BootstrapMixin):
  31. pk = forms.ModelMultipleChoiceField(queryset=VRF.objects.all(), widget=forms.MultipleHiddenInput)
  32. description = forms.CharField(max_length=100, required=False)
  33. class VRFBulkDeleteForm(ConfirmationForm):
  34. pk = forms.ModelMultipleChoiceField(queryset=VRF.objects.all(), widget=forms.MultipleHiddenInput)
  35. #
  36. # RIRs
  37. #
  38. class RIRForm(forms.ModelForm, BootstrapMixin):
  39. slug = SlugField()
  40. class Meta:
  41. model = RIR
  42. fields = ['name', 'slug']
  43. class RIRBulkDeleteForm(ConfirmationForm):
  44. pk = forms.ModelMultipleChoiceField(queryset=RIR.objects.all(), widget=forms.MultipleHiddenInput)
  45. #
  46. # Aggregates
  47. #
  48. class AggregateForm(forms.ModelForm, BootstrapMixin):
  49. class Meta:
  50. model = Aggregate
  51. fields = ['prefix', 'rir', 'date_added', 'description']
  52. help_texts = {
  53. 'prefix': "IPv4 or IPv6 network",
  54. 'rir': "Regional Internet Registry responsible for this prefix",
  55. 'date_added': "Format: YYYY-MM-DD",
  56. }
  57. class AggregateFromCSVForm(forms.ModelForm):
  58. rir = forms.ModelChoiceField(queryset=RIR.objects.all(), to_field_name='name',
  59. error_messages={'invalid_choice': 'RIR not found.'})
  60. class Meta:
  61. model = Aggregate
  62. fields = ['prefix', 'rir', 'date_added', 'description']
  63. class AggregateImportForm(BulkImportForm, BootstrapMixin):
  64. csv = CSVDataField(csv_form=AggregateFromCSVForm)
  65. class AggregateBulkEditForm(forms.Form, BootstrapMixin):
  66. pk = forms.ModelMultipleChoiceField(queryset=Aggregate.objects.all(), widget=forms.MultipleHiddenInput)
  67. rir = forms.ModelChoiceField(queryset=RIR.objects.all(), required=False, label='RIR')
  68. date_added = forms.DateField(required=False)
  69. description = forms.CharField(max_length=50, required=False)
  70. class AggregateBulkDeleteForm(ConfirmationForm):
  71. pk = forms.ModelMultipleChoiceField(queryset=Aggregate.objects.all(), widget=forms.MultipleHiddenInput)
  72. def aggregate_rir_choices():
  73. rir_choices = RIR.objects.annotate(aggregate_count=Count('aggregates'))
  74. return [(r.slug, '{} ({})'.format(r.name, r.aggregate_count)) for r in rir_choices]
  75. class AggregateFilterForm(forms.Form, BootstrapMixin):
  76. rir = forms.MultipleChoiceField(required=False, choices=aggregate_rir_choices, label='RIR',
  77. widget=forms.SelectMultiple(attrs={'size': 8}))
  78. #
  79. # Roles
  80. #
  81. class RoleForm(forms.ModelForm, BootstrapMixin):
  82. slug = SlugField()
  83. class Meta:
  84. model = Role
  85. fields = ['name', 'slug']
  86. class RoleBulkDeleteForm(ConfirmationForm):
  87. pk = forms.ModelMultipleChoiceField(queryset=Role.objects.all(), widget=forms.MultipleHiddenInput)
  88. #
  89. # Prefixes
  90. #
  91. class PrefixForm(forms.ModelForm, BootstrapMixin):
  92. site = forms.ModelChoiceField(queryset=Site.objects.all(), required=False, label='Site',
  93. widget=forms.Select(attrs={'filter-for': 'vlan'}))
  94. vlan = forms.ModelChoiceField(queryset=VLAN.objects.all(), required=False, label='VLAN',
  95. widget=APISelect(api_url='/api/ipam/vlans/?site_id={{site}}',
  96. display_field='display_name'))
  97. class Meta:
  98. model = Prefix
  99. fields = ['prefix', 'vrf', 'site', 'vlan', 'status', 'role', 'description']
  100. help_texts = {
  101. 'prefix': "IPv4 or IPv6 network",
  102. 'vrf': "VRF (if applicable)",
  103. 'site': "The site to which this prefix is assigned (if applicable)",
  104. 'vlan': "The VLAN to which this prefix is assigned (if applicable)",
  105. 'status': "Operational status of this prefix",
  106. 'role': "The primary function of this prefix",
  107. }
  108. def __init__(self, *args, **kwargs):
  109. super(PrefixForm, self).__init__(*args, **kwargs)
  110. self.fields['vrf'].empty_label = 'Global'
  111. # Initialize field without choices to avoid pulling all VLANs from the database
  112. if self.is_bound and self.data.get('site'):
  113. self.fields['vlan'].queryset = VLAN.objects.filter(site__pk=self.data['site'])
  114. elif self.initial.get('site'):
  115. self.fields['vlan'].queryset = VLAN.objects.filter(site=self.initial['site'])
  116. else:
  117. self.fields['vlan'].choices = []
  118. def clean_prefix(self):
  119. data = self.cleaned_data['prefix']
  120. try:
  121. prefix = IPNetwork(data)
  122. except:
  123. raise
  124. if prefix.version == 4 and prefix.prefixlen == 32:
  125. raise forms.ValidationError("Cannot create host addresses (/32) as prefixes. These should be IPv4 "
  126. "addresses instead.")
  127. elif prefix.version == 6 and prefix.prefixlen == 128:
  128. raise forms.ValidationError("Cannot create host addresses (/128) as prefixes. These should be IPv6 "
  129. "addresses instead.")
  130. return data
  131. class PrefixFromCSVForm(forms.ModelForm):
  132. vrf = forms.ModelChoiceField(queryset=VRF.objects.all(), required=False, to_field_name='rd',
  133. error_messages={'invalid_choice': 'VRF not found.'})
  134. site = forms.ModelChoiceField(queryset=Site.objects.all(), required=False, to_field_name='name',
  135. error_messages={'invalid_choice': 'Site not found.'})
  136. status_name = forms.ChoiceField(choices=[(s[1], s[0]) for s in PREFIX_STATUS_CHOICES])
  137. role = forms.ModelChoiceField(queryset=Role.objects.all(), required=False, to_field_name='name',
  138. error_messages={'invalid_choice': 'Invalid role.'})
  139. class Meta:
  140. model = Prefix
  141. fields = ['prefix', 'vrf', 'site', 'status_name', 'role', 'description']
  142. def save(self, *args, **kwargs):
  143. m = super(PrefixFromCSVForm, self).save(commit=False)
  144. # Assign Prefix status by name
  145. m.status = dict(self.fields['status_name'].choices)[self.cleaned_data['status_name']]
  146. if kwargs.get('commit'):
  147. m.save()
  148. return m
  149. class PrefixImportForm(BulkImportForm, BootstrapMixin):
  150. csv = CSVDataField(csv_form=PrefixFromCSVForm)
  151. class PrefixBulkEditForm(forms.Form, BootstrapMixin):
  152. pk = forms.ModelMultipleChoiceField(queryset=Prefix.objects.all(), widget=forms.MultipleHiddenInput)
  153. site = forms.ModelChoiceField(queryset=Site.objects.all(), required=False)
  154. vrf = forms.ModelChoiceField(queryset=VRF.objects.all(), required=False, label='VRF',
  155. help_text="Select the VRF to assign, or check below to remove VRF assignment")
  156. vrf_global = forms.BooleanField(required=False, label='Set VRF to global')
  157. role = forms.ModelChoiceField(queryset=Role.objects.all(), required=False)
  158. description = forms.CharField(max_length=50, required=False)
  159. class PrefixBulkDeleteForm(ConfirmationForm):
  160. pk = forms.ModelMultipleChoiceField(queryset=Prefix.objects.all(), widget=forms.MultipleHiddenInput)
  161. def prefix_vrf_choices():
  162. vrf_choices = [('', 'All'), (0, 'Global')]
  163. vrf_choices += [(v.pk, v.name) for v in VRF.objects.all()]
  164. return vrf_choices
  165. def prefix_site_choices():
  166. site_choices = Site.objects.annotate(prefix_count=Count('prefixes'))
  167. return [(s.slug, '{} ({})'.format(s.name, s.prefix_count)) for s in site_choices]
  168. def prefix_status_choices():
  169. status_counts = {}
  170. for status in Prefix.objects.values('status').annotate(count=Count('status')).order_by('status'):
  171. status_counts[status['status']] = status['count']
  172. return [(s[0], '{} ({})'.format(s[1], status_counts.get(s[0], 0))) for s in PREFIX_STATUS_CHOICES]
  173. def prefix_role_choices():
  174. role_choices = Role.objects.annotate(prefix_count=Count('prefixes'))
  175. return [(r.slug, '{} ({})'.format(r.name, r.prefix_count)) for r in role_choices]
  176. class PrefixFilterForm(forms.Form, BootstrapMixin):
  177. parent = forms.CharField(required=False, label='Search Within')
  178. vrf = forms.ChoiceField(required=False, choices=prefix_vrf_choices, label='VRF')
  179. status = forms.MultipleChoiceField(required=False, choices=prefix_status_choices)
  180. site = forms.MultipleChoiceField(required=False, choices=prefix_site_choices,
  181. widget=forms.SelectMultiple(attrs={'size': 8}))
  182. role = forms.MultipleChoiceField(required=False, choices=prefix_role_choices,
  183. widget=forms.SelectMultiple(attrs={'size': 8}))
  184. expand = forms.BooleanField(required=False, label='Expand prefix hierarchy')
  185. #
  186. # IP addresses
  187. #
  188. class IPAddressForm(forms.ModelForm, BootstrapMixin):
  189. nat_site = forms.ModelChoiceField(queryset=Site.objects.all(), required=False, label='Site',
  190. widget=forms.Select(attrs={'filter-for': 'nat_device'}))
  191. nat_device = forms.ModelChoiceField(queryset=Device.objects.all(), required=False, label='Device',
  192. widget=APISelect(api_url='/api/dcim/devices/?site_id={{nat_site}}',
  193. attrs={'filter-for': 'nat_inside'}))
  194. livesearch = forms.CharField(required=False, label='IP Address', widget=Livesearch(
  195. query_key='q', query_url='ipam-api:ipaddress_list', field_to_update='nat_inside', obj_label='address')
  196. )
  197. nat_inside = forms.ModelChoiceField(queryset=IPAddress.objects.all(), required=False, label='NAT (Inside)',
  198. widget=APISelect(api_url='/api/ipam/ip-addresses/?device_id={{nat_device}}',
  199. display_field='address'))
  200. class Meta:
  201. model = IPAddress
  202. fields = ['address', 'vrf', 'nat_device', 'nat_inside', 'description']
  203. help_texts = {
  204. 'address': "IPv4 or IPv6 address and mask",
  205. 'vrf': "VRF (if applicable)",
  206. }
  207. def __init__(self, *args, **kwargs):
  208. super(IPAddressForm, self).__init__(*args, **kwargs)
  209. self.fields['vrf'].empty_label = 'Global'
  210. if self.instance.nat_inside:
  211. nat_inside = self.instance.nat_inside
  212. # If the IP is assigned to an interface, populate site/device fields accordingly
  213. if self.instance.nat_inside.interface:
  214. self.initial['nat_site'] = self.instance.nat_inside.interface.device.rack.site.pk
  215. self.initial['nat_device'] = self.instance.nat_inside.interface.device.pk
  216. self.fields['nat_device'].queryset = Device.objects.filter(
  217. rack__site=nat_inside.interface.device.rack.site)
  218. self.fields['nat_inside'].queryset = IPAddress.objects.filter(
  219. interface__device=nat_inside.interface.device)
  220. else:
  221. self.fields['nat_inside'].queryset = IPAddress.objects.filter(pk=nat_inside.pk)
  222. else:
  223. # Initialize nat_device choices if nat_site is set
  224. if self.is_bound and self.data.get('nat_site'):
  225. self.fields['nat_device'].queryset = Device.objects.filter(rack__site__pk=self.data['nat_site'])
  226. elif self.initial.get('nat_site'):
  227. self.fields['nat_device'].queryset = Device.objects.filter(rack__site=self.initial['nat_site'])
  228. else:
  229. self.fields['nat_device'].choices = []
  230. # Initialize nat_inside choices if nat_device is set
  231. if self.is_bound and self.data.get('nat_device'):
  232. self.fields['nat_inside'].queryset = IPAddress.objects.filter(
  233. interface__device__pk=self.data['nat_device'])
  234. elif self.initial.get('nat_device'):
  235. self.fields['nat_inside'].queryset = IPAddress.objects.filter(
  236. interface__device__pk=self.initial['nat_device'])
  237. else:
  238. self.fields['nat_inside'].choices = []
  239. class IPAddressFromCSVForm(forms.ModelForm):
  240. vrf = forms.ModelChoiceField(queryset=VRF.objects.all(), required=False, to_field_name='rd',
  241. error_messages={'invalid_choice': 'Site not found.'})
  242. device = forms.ModelChoiceField(queryset=Device.objects.all(), required=False, to_field_name='name',
  243. error_messages={'invalid_choice': 'Device not found.'})
  244. interface_name = forms.CharField(required=False)
  245. is_primary = forms.BooleanField(required=False)
  246. class Meta:
  247. model = IPAddress
  248. fields = ['address', 'vrf', 'device', 'interface_name', 'is_primary', 'description']
  249. def clean(self):
  250. device = self.cleaned_data.get('device')
  251. interface_name = self.cleaned_data.get('interface_name')
  252. is_primary = self.cleaned_data.get('is_primary')
  253. # Validate interface
  254. if device and interface_name:
  255. try:
  256. Interface.objects.get(device=device, name=interface_name)
  257. except Interface.DoesNotExist:
  258. self.add_error('interface_name', "Invalid interface ({}) for {}".format(interface_name, device))
  259. elif device and not interface_name:
  260. self.add_error('interface_name', "Device set ({}) but interface missing".format(device))
  261. elif interface_name and not device:
  262. self.add_error('device', "Interface set ({}) but device missing or invalid".format(interface_name))
  263. # Validate is_primary
  264. if is_primary and not device:
  265. self.add_error('is_primary', "No device specified; cannot set as primary IP")
  266. def save(self, commit=True):
  267. # Set interface
  268. if self.cleaned_data['device'] and self.cleaned_data['interface_name']:
  269. self.instance.interface = Interface.objects.get(device=self.cleaned_data['device'],
  270. name=self.cleaned_data['interface_name'])
  271. # Set as primary for device
  272. if self.cleaned_data['is_primary']:
  273. self.instance.primary_for = self.cleaned_data['device']
  274. return super(IPAddressFromCSVForm, self).save(commit=commit)
  275. class IPAddressImportForm(BulkImportForm, BootstrapMixin):
  276. csv = CSVDataField(csv_form=IPAddressFromCSVForm)
  277. class IPAddressBulkEditForm(forms.Form, BootstrapMixin):
  278. pk = forms.ModelMultipleChoiceField(queryset=IPAddress.objects.all(), widget=forms.MultipleHiddenInput)
  279. vrf = forms.ModelChoiceField(queryset=VRF.objects.all(), required=False, label='VRF',
  280. help_text="Select the VRF to assign, or check below to remove VRF assignment")
  281. vrf_global = forms.BooleanField(required=False, label='Set VRF to global')
  282. description = forms.CharField(max_length=50, required=False)
  283. class IPAddressBulkDeleteForm(ConfirmationForm):
  284. pk = forms.ModelMultipleChoiceField(queryset=IPAddress.objects.all(), widget=forms.MultipleHiddenInput)
  285. def ipaddress_family_choices():
  286. return [('', 'All'), (4, 'IPv4'), (6, 'IPv6')]
  287. def ipaddress_vrf_choices():
  288. vrf_choices = [('', 'All'), (0, 'Global')]
  289. vrf_choices += [(v.pk, v.name) for v in VRF.objects.all()]
  290. return vrf_choices
  291. class IPAddressFilterForm(forms.Form, BootstrapMixin):
  292. family = forms.ChoiceField(required=False, choices=ipaddress_family_choices, label='Address Family')
  293. vrf = forms.ChoiceField(required=False, choices=ipaddress_vrf_choices, label='VRF')
  294. #
  295. # VLANs
  296. #
  297. class VLANForm(forms.ModelForm, BootstrapMixin):
  298. class Meta:
  299. model = VLAN
  300. fields = ['site', 'vid', 'name', 'status', 'role']
  301. help_texts = {
  302. 'site': "The site at which this VLAN exists",
  303. 'vid': "Configured VLAN ID",
  304. 'name': "Configured VLAN name",
  305. 'status': "Operational status of this VLAN",
  306. 'role': "The primary function of this VLAN",
  307. }
  308. class VLANFromCSVForm(forms.ModelForm):
  309. site = forms.ModelChoiceField(queryset=Site.objects.all(), to_field_name='name',
  310. error_messages={'invalid_choice': 'Device not found.'})
  311. status_name = forms.ChoiceField(choices=[(s[1], s[0]) for s in VLAN_STATUS_CHOICES])
  312. role = forms.ModelChoiceField(queryset=Role.objects.all(), required=False, to_field_name='name',
  313. error_messages={'invalid_choice': 'Invalid role.'})
  314. class Meta:
  315. model = VLAN
  316. fields = ['site', 'vid', 'name', 'status_name', 'role']
  317. def save(self, *args, **kwargs):
  318. m = super(VLANFromCSVForm, self).save(commit=False)
  319. # Assign VLAN status by name
  320. m.status = dict(self.fields['status_name'].choices)[self.cleaned_data['status_name']]
  321. if kwargs.get('commit'):
  322. m.save()
  323. return m
  324. class VLANImportForm(BulkImportForm, BootstrapMixin):
  325. csv = CSVDataField(csv_form=VLANFromCSVForm)
  326. class VLANBulkEditForm(forms.Form, BootstrapMixin):
  327. pk = forms.ModelMultipleChoiceField(queryset=VLAN.objects.all(), widget=forms.MultipleHiddenInput)
  328. site = forms.ModelChoiceField(queryset=Site.objects.all(), required=False)
  329. role = forms.ModelChoiceField(queryset=Role.objects.all(), required=False)
  330. class VLANBulkDeleteForm(ConfirmationForm):
  331. pk = forms.ModelMultipleChoiceField(queryset=VLAN.objects.all(), widget=forms.MultipleHiddenInput)
  332. def vlan_site_choices():
  333. site_choices = Site.objects.annotate(vlan_count=Count('vlans'))
  334. return [(s.slug, '{} ({})'.format(s.name, s.vlan_count)) for s in site_choices]
  335. def vlan_status_choices():
  336. status_counts = {}
  337. for status in VLAN.objects.values('status').annotate(count=Count('status')).order_by('status'):
  338. status_counts[status['status']] = status['count']
  339. return [(s[0], '{} ({})'.format(s[1], status_counts.get(s[0], 0))) for s in VLAN_STATUS_CHOICES]
  340. def vlan_role_choices():
  341. role_choices = Role.objects.annotate(vlan_count=Count('vlans'))
  342. return [(r.slug, '{} ({})'.format(r.name, r.vlan_count)) for r in role_choices]
  343. class VLANFilterForm(forms.Form, BootstrapMixin):
  344. site = forms.MultipleChoiceField(required=False, choices=vlan_site_choices,
  345. widget=forms.SelectMultiple(attrs={'size': 8}))
  346. status = forms.MultipleChoiceField(required=False, choices=vlan_status_choices)
  347. role = forms.MultipleChoiceField(required=False, choices=vlan_role_choices,
  348. widget=forms.SelectMultiple(attrs={'size': 8}))