forms.py 53 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880
  1. from django import forms
  2. from django.contrib.contenttypes.models import ContentType
  3. from django.utils.translation import gettext as _
  4. from dcim.models import Device, Interface, Location, Rack, Region, Site, SiteGroup
  5. from extras.forms import (
  6. AddRemoveTagsForm, CustomFieldModelBulkEditForm, CustomFieldModelCSVForm, CustomFieldModelForm, CustomFieldModelFilterForm,
  7. )
  8. from extras.models import Tag
  9. from tenancy.forms import TenancyFilterForm, TenancyForm
  10. from tenancy.models import Tenant
  11. from utilities.forms import (
  12. add_blank_choice, BootstrapMixin, BulkEditNullBooleanSelect, ContentTypeChoiceField, CSVChoiceField,
  13. CSVContentTypeField, CSVModelChoiceField, DatePicker, DynamicModelChoiceField, DynamicModelMultipleChoiceField,
  14. ExpandableIPAddressField, NumericArrayField, ReturnURLForm, SlugField, StaticSelect, StaticSelectMultiple,
  15. TagFilterField, BOOLEAN_WITH_BLANK_CHOICES,
  16. )
  17. from virtualization.models import Cluster, ClusterGroup, VirtualMachine, VMInterface
  18. from .choices import *
  19. from .constants import *
  20. from .models import *
  21. PREFIX_MASK_LENGTH_CHOICES = add_blank_choice([
  22. (i, i) for i in range(PREFIX_LENGTH_MIN, PREFIX_LENGTH_MAX + 1)
  23. ])
  24. IPADDRESS_MASK_LENGTH_CHOICES = add_blank_choice([
  25. (i, i) for i in range(IPADDRESS_MASK_LENGTH_MIN, IPADDRESS_MASK_LENGTH_MAX + 1)
  26. ])
  27. #
  28. # VRFs
  29. #
  30. class VRFForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
  31. import_targets = DynamicModelMultipleChoiceField(
  32. queryset=RouteTarget.objects.all(),
  33. required=False
  34. )
  35. export_targets = DynamicModelMultipleChoiceField(
  36. queryset=RouteTarget.objects.all(),
  37. required=False
  38. )
  39. tags = DynamicModelMultipleChoiceField(
  40. queryset=Tag.objects.all(),
  41. required=False
  42. )
  43. class Meta:
  44. model = VRF
  45. fields = [
  46. 'name', 'rd', 'enforce_unique', 'description', 'import_targets', 'export_targets', 'tenant_group', 'tenant',
  47. 'tags',
  48. ]
  49. fieldsets = (
  50. ('VRF', ('name', 'rd', 'enforce_unique', 'description', 'tags')),
  51. ('Route Targets', ('import_targets', 'export_targets')),
  52. ('Tenancy', ('tenant_group', 'tenant')),
  53. )
  54. labels = {
  55. 'rd': "RD",
  56. }
  57. help_texts = {
  58. 'rd': "Route distinguisher in any format",
  59. }
  60. class VRFCSVForm(CustomFieldModelCSVForm):
  61. tenant = CSVModelChoiceField(
  62. queryset=Tenant.objects.all(),
  63. required=False,
  64. to_field_name='name',
  65. help_text='Assigned tenant'
  66. )
  67. class Meta:
  68. model = VRF
  69. fields = ('name', 'rd', 'tenant', 'enforce_unique', 'description')
  70. class VRFBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldModelBulkEditForm):
  71. pk = forms.ModelMultipleChoiceField(
  72. queryset=VRF.objects.all(),
  73. widget=forms.MultipleHiddenInput()
  74. )
  75. tenant = DynamicModelChoiceField(
  76. queryset=Tenant.objects.all(),
  77. required=False
  78. )
  79. enforce_unique = forms.NullBooleanField(
  80. required=False,
  81. widget=BulkEditNullBooleanSelect(),
  82. label='Enforce unique space'
  83. )
  84. description = forms.CharField(
  85. max_length=100,
  86. required=False
  87. )
  88. class Meta:
  89. nullable_fields = [
  90. 'tenant', 'description',
  91. ]
  92. class VRFFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldModelFilterForm):
  93. model = VRF
  94. field_order = ['q', 'import_target_id', 'export_target_id', 'tenant_group_id', 'tenant_id']
  95. field_groups = [
  96. ['q'],
  97. ['import_target_id', 'export_target_id'],
  98. ['tenant_group_id', 'tenant_id'],
  99. ['tag']
  100. ]
  101. q = forms.CharField(
  102. required=False,
  103. widget=forms.TextInput(attrs={'placeholder': _('All Fields')}),
  104. label=_('Search')
  105. )
  106. import_target_id = DynamicModelMultipleChoiceField(
  107. queryset=RouteTarget.objects.all(),
  108. required=False,
  109. label=_('Import targets'),
  110. fetch_trigger='open'
  111. )
  112. export_target_id = DynamicModelMultipleChoiceField(
  113. queryset=RouteTarget.objects.all(),
  114. required=False,
  115. label=_('Export targets'),
  116. fetch_trigger='open'
  117. )
  118. tag = TagFilterField(model)
  119. #
  120. # Route targets
  121. #
  122. class RouteTargetForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
  123. tags = DynamicModelMultipleChoiceField(
  124. queryset=Tag.objects.all(),
  125. required=False
  126. )
  127. class Meta:
  128. model = RouteTarget
  129. fields = [
  130. 'name', 'description', 'tenant_group', 'tenant', 'tags',
  131. ]
  132. class RouteTargetCSVForm(CustomFieldModelCSVForm):
  133. tenant = CSVModelChoiceField(
  134. queryset=Tenant.objects.all(),
  135. required=False,
  136. to_field_name='name',
  137. help_text='Assigned tenant'
  138. )
  139. class Meta:
  140. model = RouteTarget
  141. fields = ('name', 'description', 'tenant')
  142. class RouteTargetBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldModelBulkEditForm):
  143. pk = forms.ModelMultipleChoiceField(
  144. queryset=RouteTarget.objects.all(),
  145. widget=forms.MultipleHiddenInput()
  146. )
  147. tenant = DynamicModelChoiceField(
  148. queryset=Tenant.objects.all(),
  149. required=False
  150. )
  151. description = forms.CharField(
  152. max_length=200,
  153. required=False
  154. )
  155. class Meta:
  156. nullable_fields = [
  157. 'tenant', 'description',
  158. ]
  159. class RouteTargetFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldModelFilterForm):
  160. model = RouteTarget
  161. field_order = ['q', 'name', 'tenant_group_id', 'tenant_id', 'importing_vrfs', 'exporting_vrfs']
  162. field_groups = [
  163. ['q'],
  164. ['importing_vrf_id', 'exporting_vrf_id'],
  165. ['tenant_group_id', 'tenant_id'],
  166. ]
  167. q = forms.CharField(
  168. required=False,
  169. widget=forms.TextInput(attrs={'placeholder': _('All Fields')}),
  170. label=_('Search')
  171. )
  172. importing_vrf_id = DynamicModelMultipleChoiceField(
  173. queryset=VRF.objects.all(),
  174. required=False,
  175. label=_('Imported by VRF'),
  176. fetch_trigger='open'
  177. )
  178. exporting_vrf_id = DynamicModelMultipleChoiceField(
  179. queryset=VRF.objects.all(),
  180. required=False,
  181. label=_('Exported by VRF'),
  182. fetch_trigger='open'
  183. )
  184. tag = TagFilterField(model)
  185. #
  186. # RIRs
  187. #
  188. class RIRForm(BootstrapMixin, CustomFieldModelForm):
  189. slug = SlugField()
  190. class Meta:
  191. model = RIR
  192. fields = [
  193. 'name', 'slug', 'is_private', 'description',
  194. ]
  195. class RIRCSVForm(CustomFieldModelCSVForm):
  196. slug = SlugField()
  197. class Meta:
  198. model = RIR
  199. fields = ('name', 'slug', 'is_private', 'description')
  200. help_texts = {
  201. 'name': 'RIR name',
  202. }
  203. class RIRBulkEditForm(BootstrapMixin, CustomFieldModelBulkEditForm):
  204. pk = forms.ModelMultipleChoiceField(
  205. queryset=RIR.objects.all(),
  206. widget=forms.MultipleHiddenInput
  207. )
  208. is_private = forms.NullBooleanField(
  209. required=False,
  210. widget=BulkEditNullBooleanSelect
  211. )
  212. description = forms.CharField(
  213. max_length=200,
  214. required=False
  215. )
  216. class Meta:
  217. nullable_fields = ['is_private', 'description']
  218. class RIRFilterForm(BootstrapMixin, forms.Form):
  219. is_private = forms.NullBooleanField(
  220. required=False,
  221. label=_('Private'),
  222. widget=StaticSelect(
  223. choices=BOOLEAN_WITH_BLANK_CHOICES
  224. )
  225. )
  226. #
  227. # Aggregates
  228. #
  229. class AggregateForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
  230. rir = DynamicModelChoiceField(
  231. queryset=RIR.objects.all(),
  232. label='RIR'
  233. )
  234. tags = DynamicModelMultipleChoiceField(
  235. queryset=Tag.objects.all(),
  236. required=False
  237. )
  238. class Meta:
  239. model = Aggregate
  240. fields = [
  241. 'prefix', 'rir', 'date_added', 'description', 'tenant_group', 'tenant', 'tags',
  242. ]
  243. fieldsets = (
  244. ('Aggregate', ('prefix', 'rir', 'date_added', 'description', 'tags')),
  245. ('Tenancy', ('tenant_group', 'tenant')),
  246. )
  247. help_texts = {
  248. 'prefix': "IPv4 or IPv6 network",
  249. 'rir': "Regional Internet Registry responsible for this prefix",
  250. }
  251. widgets = {
  252. 'date_added': DatePicker(),
  253. }
  254. class AggregateCSVForm(CustomFieldModelCSVForm):
  255. rir = CSVModelChoiceField(
  256. queryset=RIR.objects.all(),
  257. to_field_name='name',
  258. help_text='Assigned RIR'
  259. )
  260. tenant = CSVModelChoiceField(
  261. queryset=Tenant.objects.all(),
  262. required=False,
  263. to_field_name='name',
  264. help_text='Assigned tenant'
  265. )
  266. class Meta:
  267. model = Aggregate
  268. fields = ('prefix', 'rir', 'tenant', 'date_added', 'description')
  269. class AggregateBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldModelBulkEditForm):
  270. pk = forms.ModelMultipleChoiceField(
  271. queryset=Aggregate.objects.all(),
  272. widget=forms.MultipleHiddenInput()
  273. )
  274. rir = DynamicModelChoiceField(
  275. queryset=RIR.objects.all(),
  276. required=False,
  277. label='RIR'
  278. )
  279. tenant = DynamicModelChoiceField(
  280. queryset=Tenant.objects.all(),
  281. required=False
  282. )
  283. date_added = forms.DateField(
  284. required=False
  285. )
  286. description = forms.CharField(
  287. max_length=100,
  288. required=False
  289. )
  290. class Meta:
  291. nullable_fields = [
  292. 'date_added', 'description',
  293. ]
  294. widgets = {
  295. 'date_added': DatePicker(),
  296. }
  297. class AggregateFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldModelFilterForm):
  298. model = Aggregate
  299. field_order = ['q', 'family', 'rir', 'tenant_group_id', 'tenant_id']
  300. field_groups = [
  301. ['q'],
  302. ['family', 'rir_id'],
  303. ['tenant_group_id', 'tenant_id']
  304. ]
  305. q = forms.CharField(
  306. required=False,
  307. widget=forms.TextInput(attrs={'placeholder': _('All Fields')}),
  308. label=_('Search')
  309. )
  310. family = forms.ChoiceField(
  311. required=False,
  312. choices=add_blank_choice(IPAddressFamilyChoices),
  313. label=_('Address family'),
  314. widget=StaticSelect()
  315. )
  316. rir_id = DynamicModelMultipleChoiceField(
  317. queryset=RIR.objects.all(),
  318. required=False,
  319. label=_('RIR'),
  320. fetch_trigger='open'
  321. )
  322. tag = TagFilterField(model)
  323. #
  324. # Roles
  325. #
  326. class RoleForm(BootstrapMixin, CustomFieldModelForm):
  327. slug = SlugField()
  328. class Meta:
  329. model = Role
  330. fields = [
  331. 'name', 'slug', 'weight', 'description',
  332. ]
  333. class RoleCSVForm(CustomFieldModelCSVForm):
  334. slug = SlugField()
  335. class Meta:
  336. model = Role
  337. fields = ('name', 'slug', 'weight', 'description')
  338. class RoleBulkEditForm(BootstrapMixin, CustomFieldModelBulkEditForm):
  339. pk = forms.ModelMultipleChoiceField(
  340. queryset=Role.objects.all(),
  341. widget=forms.MultipleHiddenInput
  342. )
  343. weight = forms.IntegerField(
  344. required=False
  345. )
  346. description = forms.CharField(
  347. max_length=200,
  348. required=False
  349. )
  350. class Meta:
  351. nullable_fields = ['description']
  352. #
  353. # Prefixes
  354. #
  355. class PrefixForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
  356. vrf = DynamicModelChoiceField(
  357. queryset=VRF.objects.all(),
  358. required=False,
  359. label='VRF'
  360. )
  361. region = DynamicModelChoiceField(
  362. queryset=Region.objects.all(),
  363. required=False,
  364. initial_params={
  365. 'sites': '$site'
  366. }
  367. )
  368. site_group = DynamicModelChoiceField(
  369. queryset=SiteGroup.objects.all(),
  370. required=False,
  371. initial_params={
  372. 'sites': '$site'
  373. }
  374. )
  375. site = DynamicModelChoiceField(
  376. queryset=Site.objects.all(),
  377. required=False,
  378. null_option='None',
  379. query_params={
  380. 'region_id': '$region',
  381. 'group_id': '$site_group',
  382. }
  383. )
  384. vlan_group = DynamicModelChoiceField(
  385. queryset=VLANGroup.objects.all(),
  386. required=False,
  387. label='VLAN group',
  388. null_option='None',
  389. query_params={
  390. 'site_id': '$site'
  391. },
  392. initial_params={
  393. 'vlans': '$vlan'
  394. }
  395. )
  396. vlan = DynamicModelChoiceField(
  397. queryset=VLAN.objects.all(),
  398. required=False,
  399. label='VLAN',
  400. query_params={
  401. 'site_id': '$site',
  402. 'group_id': '$vlan_group',
  403. }
  404. )
  405. role = DynamicModelChoiceField(
  406. queryset=Role.objects.all(),
  407. required=False
  408. )
  409. tags = DynamicModelMultipleChoiceField(
  410. queryset=Tag.objects.all(),
  411. required=False
  412. )
  413. class Meta:
  414. model = Prefix
  415. fields = [
  416. 'prefix', 'vrf', 'site', 'vlan', 'status', 'role', 'is_pool', 'mark_utilized', 'description',
  417. 'tenant_group', 'tenant', 'tags',
  418. ]
  419. fieldsets = (
  420. ('Prefix', ('prefix', 'status', 'vrf', 'role', 'is_pool', 'mark_utilized', 'description', 'tags')),
  421. ('Site/VLAN Assignment', ('region', 'site_group', 'site', 'vlan_group', 'vlan')),
  422. ('Tenancy', ('tenant_group', 'tenant')),
  423. )
  424. widgets = {
  425. 'status': StaticSelect(),
  426. }
  427. def __init__(self, *args, **kwargs):
  428. super().__init__(*args, **kwargs)
  429. self.fields['vrf'].empty_label = 'Global'
  430. class PrefixCSVForm(CustomFieldModelCSVForm):
  431. vrf = CSVModelChoiceField(
  432. queryset=VRF.objects.all(),
  433. to_field_name='name',
  434. required=False,
  435. help_text='Assigned VRF'
  436. )
  437. tenant = CSVModelChoiceField(
  438. queryset=Tenant.objects.all(),
  439. required=False,
  440. to_field_name='name',
  441. help_text='Assigned tenant'
  442. )
  443. site = CSVModelChoiceField(
  444. queryset=Site.objects.all(),
  445. required=False,
  446. to_field_name='name',
  447. help_text='Assigned site'
  448. )
  449. vlan_group = CSVModelChoiceField(
  450. queryset=VLANGroup.objects.all(),
  451. required=False,
  452. to_field_name='name',
  453. help_text="VLAN's group (if any)"
  454. )
  455. vlan = CSVModelChoiceField(
  456. queryset=VLAN.objects.all(),
  457. required=False,
  458. to_field_name='vid',
  459. help_text="Assigned VLAN"
  460. )
  461. status = CSVChoiceField(
  462. choices=PrefixStatusChoices,
  463. help_text='Operational status'
  464. )
  465. role = CSVModelChoiceField(
  466. queryset=Role.objects.all(),
  467. required=False,
  468. to_field_name='name',
  469. help_text='Functional role'
  470. )
  471. class Meta:
  472. model = Prefix
  473. fields = (
  474. 'prefix', 'vrf', 'tenant', 'site', 'vlan_group', 'vlan', 'status', 'role', 'is_pool', 'mark_utilized',
  475. 'description',
  476. )
  477. def __init__(self, data=None, *args, **kwargs):
  478. super().__init__(data, *args, **kwargs)
  479. if data:
  480. # Limit VLAN queryset by assigned site and/or group (if specified)
  481. params = {}
  482. if data.get('site'):
  483. params[f"site__{self.fields['site'].to_field_name}"] = data.get('site')
  484. if data.get('vlan_group'):
  485. params[f"group__{self.fields['vlan_group'].to_field_name}"] = data.get('vlan_group')
  486. if params:
  487. self.fields['vlan'].queryset = self.fields['vlan'].queryset.filter(**params)
  488. class PrefixBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldModelBulkEditForm):
  489. pk = forms.ModelMultipleChoiceField(
  490. queryset=Prefix.objects.all(),
  491. widget=forms.MultipleHiddenInput()
  492. )
  493. region = DynamicModelChoiceField(
  494. queryset=Region.objects.all(),
  495. required=False
  496. )
  497. site_group = DynamicModelChoiceField(
  498. queryset=SiteGroup.objects.all(),
  499. required=False
  500. )
  501. site = DynamicModelChoiceField(
  502. queryset=Site.objects.all(),
  503. required=False,
  504. query_params={
  505. 'region_id': '$region',
  506. 'group_id': '$site_group',
  507. }
  508. )
  509. vrf = DynamicModelChoiceField(
  510. queryset=VRF.objects.all(),
  511. required=False,
  512. label='VRF'
  513. )
  514. prefix_length = forms.IntegerField(
  515. min_value=PREFIX_LENGTH_MIN,
  516. max_value=PREFIX_LENGTH_MAX,
  517. required=False
  518. )
  519. tenant = DynamicModelChoiceField(
  520. queryset=Tenant.objects.all(),
  521. required=False
  522. )
  523. status = forms.ChoiceField(
  524. choices=add_blank_choice(PrefixStatusChoices),
  525. required=False,
  526. widget=StaticSelect()
  527. )
  528. role = DynamicModelChoiceField(
  529. queryset=Role.objects.all(),
  530. required=False
  531. )
  532. is_pool = forms.NullBooleanField(
  533. required=False,
  534. widget=BulkEditNullBooleanSelect(),
  535. label='Is a pool'
  536. )
  537. mark_utilized = forms.NullBooleanField(
  538. required=False,
  539. widget=BulkEditNullBooleanSelect(),
  540. label='Treat as 100% utilized'
  541. )
  542. description = forms.CharField(
  543. max_length=100,
  544. required=False
  545. )
  546. class Meta:
  547. nullable_fields = [
  548. 'site', 'vrf', 'tenant', 'role', 'description',
  549. ]
  550. class PrefixFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldModelFilterForm):
  551. model = Prefix
  552. field_order = [
  553. 'q', 'within_include', 'family', 'mask_length', 'vrf_id', 'present_in_vrf_id', 'status',
  554. 'region_id', 'site_group_id', 'site_id', 'role_id', 'tenant_group_id', 'tenant_id',
  555. 'is_pool', 'mark_utilized',
  556. ]
  557. field_groups = [
  558. ['q'],
  559. ['role_id', 'within_include', 'family', 'mask_length'],
  560. ['vrf_id', 'present_in_vrf_id', 'is_pool', 'mark_utilized'],
  561. ['region_id', 'site_group_id', 'site_id'],
  562. ['tenant_group_id', 'tenant_id', 'status', 'tag']
  563. ]
  564. q = forms.CharField(
  565. required=False,
  566. widget=forms.TextInput(attrs={'placeholder': _('All Fields')}),
  567. label=_('Search')
  568. )
  569. mask_length__lte = forms.IntegerField(
  570. widget=forms.HiddenInput()
  571. )
  572. within_include = forms.CharField(
  573. required=False,
  574. widget=forms.TextInput(
  575. attrs={
  576. 'placeholder': 'Prefix',
  577. }
  578. ),
  579. label=_('Search within')
  580. )
  581. family = forms.ChoiceField(
  582. required=False,
  583. choices=add_blank_choice(IPAddressFamilyChoices),
  584. label=_('Address family'),
  585. widget=StaticSelect()
  586. )
  587. mask_length = forms.ChoiceField(
  588. required=False,
  589. choices=PREFIX_MASK_LENGTH_CHOICES,
  590. label=_('Mask length'),
  591. widget=StaticSelect()
  592. )
  593. vrf_id = DynamicModelMultipleChoiceField(
  594. queryset=VRF.objects.all(),
  595. required=False,
  596. label=_('Assigned VRF'),
  597. null_option='Global',
  598. fetch_trigger='open'
  599. )
  600. present_in_vrf_id = DynamicModelChoiceField(
  601. queryset=VRF.objects.all(),
  602. required=False,
  603. label=_('Present in VRF'),
  604. fetch_trigger='open'
  605. )
  606. status = forms.MultipleChoiceField(
  607. choices=PrefixStatusChoices,
  608. required=False,
  609. widget=StaticSelectMultiple()
  610. )
  611. region_id = DynamicModelMultipleChoiceField(
  612. queryset=Region.objects.all(),
  613. required=False,
  614. label=_('Region'),
  615. fetch_trigger='open'
  616. )
  617. site_group_id = DynamicModelMultipleChoiceField(
  618. queryset=SiteGroup.objects.all(),
  619. required=False,
  620. label=_('Site group'),
  621. fetch_trigger='open'
  622. )
  623. site_id = DynamicModelMultipleChoiceField(
  624. queryset=Site.objects.all(),
  625. required=False,
  626. null_option='None',
  627. query_params={
  628. 'region_id': '$region_id'
  629. },
  630. label=_('Site'),
  631. fetch_trigger='open'
  632. )
  633. role_id = DynamicModelMultipleChoiceField(
  634. queryset=Role.objects.all(),
  635. required=False,
  636. null_option='None',
  637. label=_('Role'),
  638. fetch_trigger='open'
  639. )
  640. is_pool = forms.NullBooleanField(
  641. required=False,
  642. label=_('Is a pool'),
  643. widget=StaticSelect(
  644. choices=BOOLEAN_WITH_BLANK_CHOICES
  645. )
  646. )
  647. mark_utilized = forms.NullBooleanField(
  648. required=False,
  649. label=_('Marked as 100% utilized'),
  650. widget=StaticSelect(
  651. choices=BOOLEAN_WITH_BLANK_CHOICES
  652. )
  653. )
  654. tag = TagFilterField(model)
  655. #
  656. # IP ranges
  657. #
  658. class IPRangeForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
  659. vrf = DynamicModelChoiceField(
  660. queryset=VRF.objects.all(),
  661. required=False,
  662. label='VRF'
  663. )
  664. role = DynamicModelChoiceField(
  665. queryset=Role.objects.all(),
  666. required=False
  667. )
  668. tags = DynamicModelMultipleChoiceField(
  669. queryset=Tag.objects.all(),
  670. required=False
  671. )
  672. class Meta:
  673. model = IPRange
  674. fields = [
  675. 'vrf', 'start_address', 'end_address', 'status', 'role', 'description', 'tenant_group', 'tenant', 'tags',
  676. ]
  677. fieldsets = (
  678. ('IP Range', ('vrf', 'start_address', 'end_address', 'role', 'status', 'description', 'tags')),
  679. ('Tenancy', ('tenant_group', 'tenant')),
  680. )
  681. widgets = {
  682. 'status': StaticSelect(),
  683. }
  684. def __init__(self, *args, **kwargs):
  685. super().__init__(*args, **kwargs)
  686. self.fields['vrf'].empty_label = 'Global'
  687. class IPRangeCSVForm(CustomFieldModelCSVForm):
  688. vrf = CSVModelChoiceField(
  689. queryset=VRF.objects.all(),
  690. to_field_name='name',
  691. required=False,
  692. help_text='Assigned VRF'
  693. )
  694. tenant = CSVModelChoiceField(
  695. queryset=Tenant.objects.all(),
  696. required=False,
  697. to_field_name='name',
  698. help_text='Assigned tenant'
  699. )
  700. status = CSVChoiceField(
  701. choices=IPRangeStatusChoices,
  702. help_text='Operational status'
  703. )
  704. role = CSVModelChoiceField(
  705. queryset=Role.objects.all(),
  706. required=False,
  707. to_field_name='name',
  708. help_text='Functional role'
  709. )
  710. class Meta:
  711. model = IPRange
  712. fields = (
  713. 'start_address', 'end_address', 'vrf', 'tenant', 'status', 'role', 'description',
  714. )
  715. class IPRangeBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldModelBulkEditForm):
  716. pk = forms.ModelMultipleChoiceField(
  717. queryset=IPRange.objects.all(),
  718. widget=forms.MultipleHiddenInput()
  719. )
  720. vrf = DynamicModelChoiceField(
  721. queryset=VRF.objects.all(),
  722. required=False,
  723. label='VRF'
  724. )
  725. tenant = DynamicModelChoiceField(
  726. queryset=Tenant.objects.all(),
  727. required=False
  728. )
  729. status = forms.ChoiceField(
  730. choices=add_blank_choice(IPRangeStatusChoices),
  731. required=False,
  732. widget=StaticSelect()
  733. )
  734. role = DynamicModelChoiceField(
  735. queryset=Role.objects.all(),
  736. required=False
  737. )
  738. description = forms.CharField(
  739. max_length=100,
  740. required=False
  741. )
  742. class Meta:
  743. nullable_fields = [
  744. 'vrf', 'tenant', 'role', 'description',
  745. ]
  746. class IPRangeFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldModelFilterForm):
  747. model = IPRange
  748. field_order = [
  749. 'q', 'family', 'vrf_id', 'status', 'role_id', 'tenant_group_id', 'tenant_id',
  750. ]
  751. field_groups = [
  752. ['q'],
  753. ['family', 'vrf_id', 'status', 'role_id'],
  754. ['tenant_group_id', 'tenant_id', 'tag'],
  755. ]
  756. q = forms.CharField(
  757. required=False,
  758. widget=forms.TextInput(attrs={'placeholder': _('All Fields')}),
  759. label=_('Search')
  760. )
  761. family = forms.ChoiceField(
  762. required=False,
  763. choices=add_blank_choice(IPAddressFamilyChoices),
  764. label=_('Address family'),
  765. widget=StaticSelect()
  766. )
  767. vrf_id = DynamicModelMultipleChoiceField(
  768. queryset=VRF.objects.all(),
  769. required=False,
  770. label=_('Assigned VRF'),
  771. null_option='Global',
  772. fetch_trigger='open'
  773. )
  774. status = forms.MultipleChoiceField(
  775. choices=PrefixStatusChoices,
  776. required=False,
  777. widget=StaticSelectMultiple()
  778. )
  779. role_id = DynamicModelMultipleChoiceField(
  780. queryset=Role.objects.all(),
  781. required=False,
  782. null_option='None',
  783. label=_('Role'),
  784. fetch_trigger='open'
  785. )
  786. tag = TagFilterField(model)
  787. #
  788. # IP addresses
  789. #
  790. class IPAddressForm(BootstrapMixin, TenancyForm, ReturnURLForm, CustomFieldModelForm):
  791. device = DynamicModelChoiceField(
  792. queryset=Device.objects.all(),
  793. required=False,
  794. initial_params={
  795. 'interfaces': '$interface'
  796. }
  797. )
  798. interface = DynamicModelChoiceField(
  799. queryset=Interface.objects.all(),
  800. required=False,
  801. query_params={
  802. 'device_id': '$device'
  803. }
  804. )
  805. virtual_machine = DynamicModelChoiceField(
  806. queryset=VirtualMachine.objects.all(),
  807. required=False,
  808. initial_params={
  809. 'interfaces': '$vminterface'
  810. }
  811. )
  812. vminterface = DynamicModelChoiceField(
  813. queryset=VMInterface.objects.all(),
  814. required=False,
  815. label='Interface',
  816. query_params={
  817. 'virtual_machine_id': '$virtual_machine'
  818. }
  819. )
  820. vrf = DynamicModelChoiceField(
  821. queryset=VRF.objects.all(),
  822. required=False,
  823. label='VRF'
  824. )
  825. nat_region = DynamicModelChoiceField(
  826. queryset=Region.objects.all(),
  827. required=False,
  828. label='Region',
  829. initial_params={
  830. 'sites': '$nat_site'
  831. }
  832. )
  833. nat_site_group = DynamicModelChoiceField(
  834. queryset=SiteGroup.objects.all(),
  835. required=False,
  836. label='Site group',
  837. initial_params={
  838. 'sites': '$nat_site'
  839. }
  840. )
  841. nat_site = DynamicModelChoiceField(
  842. queryset=Site.objects.all(),
  843. required=False,
  844. label='Site',
  845. query_params={
  846. 'region_id': '$nat_region',
  847. 'group_id': '$nat_site_group',
  848. }
  849. )
  850. nat_rack = DynamicModelChoiceField(
  851. queryset=Rack.objects.all(),
  852. required=False,
  853. label='Rack',
  854. null_option='None',
  855. query_params={
  856. 'site_id': '$site'
  857. }
  858. )
  859. nat_device = DynamicModelChoiceField(
  860. queryset=Device.objects.all(),
  861. required=False,
  862. label='Device',
  863. query_params={
  864. 'site_id': '$site',
  865. 'rack_id': '$nat_rack',
  866. }
  867. )
  868. nat_cluster = DynamicModelChoiceField(
  869. queryset=Cluster.objects.all(),
  870. required=False,
  871. label='Cluster'
  872. )
  873. nat_virtual_machine = DynamicModelChoiceField(
  874. queryset=VirtualMachine.objects.all(),
  875. required=False,
  876. label='Virtual Machine',
  877. query_params={
  878. 'cluster_id': '$nat_cluster',
  879. }
  880. )
  881. nat_vrf = DynamicModelChoiceField(
  882. queryset=VRF.objects.all(),
  883. required=False,
  884. label='VRF'
  885. )
  886. nat_inside = DynamicModelChoiceField(
  887. queryset=IPAddress.objects.all(),
  888. required=False,
  889. label='IP Address',
  890. query_params={
  891. 'device_id': '$nat_device',
  892. 'virtual_machine_id': '$nat_virtual_machine',
  893. 'vrf_id': '$nat_vrf',
  894. }
  895. )
  896. primary_for_parent = forms.BooleanField(
  897. required=False,
  898. label='Make this the primary IP for the device/VM'
  899. )
  900. tags = DynamicModelMultipleChoiceField(
  901. queryset=Tag.objects.all(),
  902. required=False
  903. )
  904. class Meta:
  905. model = IPAddress
  906. fields = [
  907. 'address', 'vrf', 'status', 'role', 'dns_name', 'description', 'primary_for_parent', 'nat_site', 'nat_rack',
  908. 'nat_device', 'nat_cluster', 'nat_virtual_machine', 'nat_vrf', 'nat_inside', 'tenant_group', 'tenant',
  909. 'tags',
  910. ]
  911. widgets = {
  912. 'status': StaticSelect(),
  913. 'role': StaticSelect(),
  914. }
  915. def __init__(self, *args, **kwargs):
  916. # Initialize helper selectors
  917. instance = kwargs.get('instance')
  918. initial = kwargs.get('initial', {}).copy()
  919. if instance:
  920. if type(instance.assigned_object) is Interface:
  921. initial['interface'] = instance.assigned_object
  922. elif type(instance.assigned_object) is VMInterface:
  923. initial['vminterface'] = instance.assigned_object
  924. if instance.nat_inside:
  925. nat_inside_parent = instance.nat_inside.assigned_object
  926. if type(nat_inside_parent) is Interface:
  927. initial['nat_site'] = nat_inside_parent.device.site.pk
  928. if nat_inside_parent.device.rack:
  929. initial['nat_rack'] = nat_inside_parent.device.rack.pk
  930. initial['nat_device'] = nat_inside_parent.device.pk
  931. elif type(nat_inside_parent) is VMInterface:
  932. initial['nat_cluster'] = nat_inside_parent.virtual_machine.cluster.pk
  933. initial['nat_virtual_machine'] = nat_inside_parent.virtual_machine.pk
  934. kwargs['initial'] = initial
  935. super().__init__(*args, **kwargs)
  936. self.fields['vrf'].empty_label = 'Global'
  937. # Initialize primary_for_parent if IP address is already assigned
  938. if self.instance.pk and self.instance.assigned_object:
  939. parent = self.instance.assigned_object.parent_object
  940. if (
  941. self.instance.address.version == 4 and parent.primary_ip4_id == self.instance.pk or
  942. self.instance.address.version == 6 and parent.primary_ip6_id == self.instance.pk
  943. ):
  944. self.initial['primary_for_parent'] = True
  945. def clean(self):
  946. super().clean()
  947. # Cannot select both a device interface and a VM interface
  948. if self.cleaned_data.get('interface') and self.cleaned_data.get('vminterface'):
  949. raise forms.ValidationError("Cannot select both a device interface and a virtual machine interface")
  950. self.instance.assigned_object = self.cleaned_data.get('interface') or self.cleaned_data.get('vminterface')
  951. # Primary IP assignment is only available if an interface has been assigned.
  952. interface = self.cleaned_data.get('interface') or self.cleaned_data.get('vminterface')
  953. if self.cleaned_data.get('primary_for_parent') and not interface:
  954. self.add_error(
  955. 'primary_for_parent', "Only IP addresses assigned to an interface can be designated as primary IPs."
  956. )
  957. def save(self, *args, **kwargs):
  958. ipaddress = super().save(*args, **kwargs)
  959. # Assign/clear this IPAddress as the primary for the associated Device/VirtualMachine.
  960. interface = self.instance.assigned_object
  961. if interface:
  962. parent = interface.parent_object
  963. if self.cleaned_data['primary_for_parent']:
  964. if ipaddress.address.version == 4:
  965. parent.primary_ip4 = ipaddress
  966. else:
  967. parent.primary_ip6 = ipaddress
  968. parent.save()
  969. elif ipaddress.address.version == 4 and parent.primary_ip4 == ipaddress:
  970. parent.primary_ip4 = None
  971. parent.save()
  972. elif ipaddress.address.version == 6 and parent.primary_ip6 == ipaddress:
  973. parent.primary_ip6 = None
  974. parent.save()
  975. return ipaddress
  976. class IPAddressBulkCreateForm(BootstrapMixin, forms.Form):
  977. pattern = ExpandableIPAddressField(
  978. label='Address pattern'
  979. )
  980. class IPAddressBulkAddForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
  981. vrf = DynamicModelChoiceField(
  982. queryset=VRF.objects.all(),
  983. required=False,
  984. label='VRF'
  985. )
  986. tags = DynamicModelMultipleChoiceField(
  987. queryset=Tag.objects.all(),
  988. required=False
  989. )
  990. class Meta:
  991. model = IPAddress
  992. fields = [
  993. 'address', 'vrf', 'status', 'role', 'dns_name', 'description', 'tenant_group', 'tenant', 'tags',
  994. ]
  995. widgets = {
  996. 'status': StaticSelect(),
  997. 'role': StaticSelect(),
  998. }
  999. def __init__(self, *args, **kwargs):
  1000. super().__init__(*args, **kwargs)
  1001. self.fields['vrf'].empty_label = 'Global'
  1002. class IPAddressCSVForm(CustomFieldModelCSVForm):
  1003. vrf = CSVModelChoiceField(
  1004. queryset=VRF.objects.all(),
  1005. to_field_name='name',
  1006. required=False,
  1007. help_text='Assigned VRF'
  1008. )
  1009. tenant = CSVModelChoiceField(
  1010. queryset=Tenant.objects.all(),
  1011. to_field_name='name',
  1012. required=False,
  1013. help_text='Assigned tenant'
  1014. )
  1015. status = CSVChoiceField(
  1016. choices=IPAddressStatusChoices,
  1017. required=False,
  1018. help_text='Operational status'
  1019. )
  1020. role = CSVChoiceField(
  1021. choices=IPAddressRoleChoices,
  1022. required=False,
  1023. help_text='Functional role'
  1024. )
  1025. device = CSVModelChoiceField(
  1026. queryset=Device.objects.all(),
  1027. required=False,
  1028. to_field_name='name',
  1029. help_text='Parent device of assigned interface (if any)'
  1030. )
  1031. virtual_machine = CSVModelChoiceField(
  1032. queryset=VirtualMachine.objects.all(),
  1033. required=False,
  1034. to_field_name='name',
  1035. help_text='Parent VM of assigned interface (if any)'
  1036. )
  1037. interface = CSVModelChoiceField(
  1038. queryset=Interface.objects.none(), # Can also refer to VMInterface
  1039. required=False,
  1040. to_field_name='name',
  1041. help_text='Assigned interface'
  1042. )
  1043. is_primary = forms.BooleanField(
  1044. help_text='Make this the primary IP for the assigned device',
  1045. required=False
  1046. )
  1047. class Meta:
  1048. model = IPAddress
  1049. fields = [
  1050. 'address', 'vrf', 'tenant', 'status', 'role', 'device', 'virtual_machine', 'interface', 'is_primary',
  1051. 'dns_name', 'description',
  1052. ]
  1053. def __init__(self, data=None, *args, **kwargs):
  1054. super().__init__(data, *args, **kwargs)
  1055. if data:
  1056. # Limit interface queryset by assigned device
  1057. if data.get('device'):
  1058. self.fields['interface'].queryset = Interface.objects.filter(
  1059. **{f"device__{self.fields['device'].to_field_name}": data['device']}
  1060. )
  1061. # Limit interface queryset by assigned device
  1062. elif data.get('virtual_machine'):
  1063. self.fields['interface'].queryset = VMInterface.objects.filter(
  1064. **{f"virtual_machine__{self.fields['virtual_machine'].to_field_name}": data['virtual_machine']}
  1065. )
  1066. def clean(self):
  1067. super().clean()
  1068. device = self.cleaned_data.get('device')
  1069. virtual_machine = self.cleaned_data.get('virtual_machine')
  1070. is_primary = self.cleaned_data.get('is_primary')
  1071. # Validate is_primary
  1072. if is_primary and not device and not virtual_machine:
  1073. raise forms.ValidationError("No device or virtual machine specified; cannot set as primary IP")
  1074. def save(self, *args, **kwargs):
  1075. # Set interface assignment
  1076. if self.cleaned_data['interface']:
  1077. self.instance.assigned_object = self.cleaned_data['interface']
  1078. ipaddress = super().save(*args, **kwargs)
  1079. # Set as primary for device/VM
  1080. if self.cleaned_data['is_primary']:
  1081. parent = self.cleaned_data['device'] or self.cleaned_data['virtual_machine']
  1082. if self.instance.address.version == 4:
  1083. parent.primary_ip4 = ipaddress
  1084. elif self.instance.address.version == 6:
  1085. parent.primary_ip6 = ipaddress
  1086. parent.save()
  1087. return ipaddress
  1088. class IPAddressBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldModelBulkEditForm):
  1089. pk = forms.ModelMultipleChoiceField(
  1090. queryset=IPAddress.objects.all(),
  1091. widget=forms.MultipleHiddenInput()
  1092. )
  1093. vrf = DynamicModelChoiceField(
  1094. queryset=VRF.objects.all(),
  1095. required=False,
  1096. label='VRF'
  1097. )
  1098. mask_length = forms.IntegerField(
  1099. min_value=IPADDRESS_MASK_LENGTH_MIN,
  1100. max_value=IPADDRESS_MASK_LENGTH_MAX,
  1101. required=False
  1102. )
  1103. tenant = DynamicModelChoiceField(
  1104. queryset=Tenant.objects.all(),
  1105. required=False
  1106. )
  1107. status = forms.ChoiceField(
  1108. choices=add_blank_choice(IPAddressStatusChoices),
  1109. required=False,
  1110. widget=StaticSelect()
  1111. )
  1112. role = forms.ChoiceField(
  1113. choices=add_blank_choice(IPAddressRoleChoices),
  1114. required=False,
  1115. widget=StaticSelect()
  1116. )
  1117. dns_name = forms.CharField(
  1118. max_length=255,
  1119. required=False
  1120. )
  1121. description = forms.CharField(
  1122. max_length=100,
  1123. required=False
  1124. )
  1125. class Meta:
  1126. nullable_fields = [
  1127. 'vrf', 'role', 'tenant', 'dns_name', 'description',
  1128. ]
  1129. class IPAddressAssignForm(BootstrapMixin, forms.Form):
  1130. vrf_id = DynamicModelChoiceField(
  1131. queryset=VRF.objects.all(),
  1132. required=False,
  1133. label='VRF',
  1134. empty_label='Global'
  1135. )
  1136. q = forms.CharField(
  1137. required=False,
  1138. label='Search',
  1139. )
  1140. class IPAddressFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldModelFilterForm):
  1141. model = IPAddress
  1142. field_order = [
  1143. 'q', 'parent', 'family', 'mask_length', 'vrf_id', 'present_in_vrf_id', 'status', 'role',
  1144. 'assigned_to_interface', 'tenant_group_id', 'tenant_id',
  1145. ]
  1146. field_groups = [
  1147. ['q'],
  1148. ['parent', 'family', 'mask_length'],
  1149. ['status', 'vrf_id', 'present_in_vrf_id'],
  1150. ['role', 'assigned_to_interface'],
  1151. ['tenant_group_id', 'tenant_id'],
  1152. ]
  1153. q = forms.CharField(
  1154. required=False,
  1155. widget=forms.TextInput(attrs={'placeholder': _('All Fields')}),
  1156. label=_('Search')
  1157. )
  1158. parent = forms.CharField(
  1159. required=False,
  1160. widget=forms.TextInput(
  1161. attrs={
  1162. 'placeholder': 'Prefix',
  1163. }
  1164. ),
  1165. label='Parent Prefix'
  1166. )
  1167. family = forms.ChoiceField(
  1168. required=False,
  1169. choices=add_blank_choice(IPAddressFamilyChoices),
  1170. label=_('Address family'),
  1171. widget=StaticSelect()
  1172. )
  1173. mask_length = forms.ChoiceField(
  1174. required=False,
  1175. choices=IPADDRESS_MASK_LENGTH_CHOICES,
  1176. label=_('Mask length'),
  1177. widget=StaticSelect()
  1178. )
  1179. vrf_id = DynamicModelMultipleChoiceField(
  1180. queryset=VRF.objects.all(),
  1181. required=False,
  1182. label=_('Assigned VRF'),
  1183. null_option='Global',
  1184. fetch_trigger='open'
  1185. )
  1186. present_in_vrf_id = DynamicModelChoiceField(
  1187. queryset=VRF.objects.all(),
  1188. required=False,
  1189. label=_('Present in VRF'),
  1190. fetch_trigger='open'
  1191. )
  1192. status = forms.MultipleChoiceField(
  1193. choices=IPAddressStatusChoices,
  1194. required=False,
  1195. widget=StaticSelectMultiple()
  1196. )
  1197. role = forms.MultipleChoiceField(
  1198. choices=IPAddressRoleChoices,
  1199. required=False,
  1200. widget=StaticSelectMultiple()
  1201. )
  1202. assigned_to_interface = forms.NullBooleanField(
  1203. required=False,
  1204. label=_('Assigned to an interface'),
  1205. widget=StaticSelect(
  1206. choices=BOOLEAN_WITH_BLANK_CHOICES
  1207. )
  1208. )
  1209. tag = TagFilterField(model)
  1210. #
  1211. # VLAN groups
  1212. #
  1213. class VLANGroupForm(BootstrapMixin, CustomFieldModelForm):
  1214. scope_type = ContentTypeChoiceField(
  1215. queryset=ContentType.objects.filter(model__in=VLANGROUP_SCOPE_TYPES),
  1216. required=False,
  1217. widget=StaticSelect
  1218. )
  1219. region = DynamicModelChoiceField(
  1220. queryset=Region.objects.all(),
  1221. required=False,
  1222. initial_params={
  1223. 'sites': '$site'
  1224. }
  1225. )
  1226. sitegroup = DynamicModelChoiceField(
  1227. queryset=SiteGroup.objects.all(),
  1228. required=False,
  1229. initial_params={
  1230. 'sites': '$site'
  1231. },
  1232. label='Site group'
  1233. )
  1234. site = DynamicModelChoiceField(
  1235. queryset=Site.objects.all(),
  1236. required=False,
  1237. initial_params={
  1238. 'locations': '$location'
  1239. },
  1240. query_params={
  1241. 'region_id': '$region',
  1242. 'group_id': '$sitegroup',
  1243. }
  1244. )
  1245. location = DynamicModelChoiceField(
  1246. queryset=Location.objects.all(),
  1247. required=False,
  1248. initial_params={
  1249. 'racks': '$rack'
  1250. },
  1251. query_params={
  1252. 'site_id': '$site',
  1253. }
  1254. )
  1255. rack = DynamicModelChoiceField(
  1256. queryset=Rack.objects.all(),
  1257. required=False,
  1258. query_params={
  1259. 'site_id': '$site',
  1260. 'location_id': '$location',
  1261. }
  1262. )
  1263. clustergroup = DynamicModelChoiceField(
  1264. queryset=ClusterGroup.objects.all(),
  1265. required=False,
  1266. initial_params={
  1267. 'clusters': '$cluster'
  1268. },
  1269. label='Cluster group'
  1270. )
  1271. cluster = DynamicModelChoiceField(
  1272. queryset=Cluster.objects.all(),
  1273. required=False,
  1274. query_params={
  1275. 'group_id': '$clustergroup',
  1276. }
  1277. )
  1278. slug = SlugField()
  1279. class Meta:
  1280. model = VLANGroup
  1281. fields = [
  1282. 'name', 'slug', 'description', 'scope_type', 'region', 'sitegroup', 'site', 'location', 'rack',
  1283. 'clustergroup', 'cluster',
  1284. ]
  1285. widgets = {
  1286. 'scope_type': StaticSelect,
  1287. }
  1288. def __init__(self, *args, **kwargs):
  1289. instance = kwargs.get('instance')
  1290. initial = kwargs.get('initial', {})
  1291. if instance is not None and instance.scope:
  1292. initial[instance.scope_type.model] = instance.scope
  1293. kwargs['initial'] = initial
  1294. super().__init__(*args, **kwargs)
  1295. def clean(self):
  1296. super().clean()
  1297. # Assign scope based on scope_type
  1298. if self.cleaned_data.get('scope_type'):
  1299. scope_field = self.cleaned_data['scope_type'].model
  1300. self.instance.scope = self.cleaned_data.get(scope_field)
  1301. else:
  1302. self.instance.scope_id = None
  1303. class VLANGroupCSVForm(CustomFieldModelCSVForm):
  1304. slug = SlugField()
  1305. scope_type = CSVContentTypeField(
  1306. queryset=ContentType.objects.filter(model__in=VLANGROUP_SCOPE_TYPES),
  1307. required=False,
  1308. label='Scope type (app & model)'
  1309. )
  1310. class Meta:
  1311. model = VLANGroup
  1312. fields = ('name', 'slug', 'scope_type', 'scope_id', 'description')
  1313. labels = {
  1314. 'scope_id': 'Scope ID',
  1315. }
  1316. class VLANGroupBulkEditForm(BootstrapMixin, CustomFieldModelBulkEditForm):
  1317. pk = forms.ModelMultipleChoiceField(
  1318. queryset=VLANGroup.objects.all(),
  1319. widget=forms.MultipleHiddenInput
  1320. )
  1321. site = DynamicModelChoiceField(
  1322. queryset=Site.objects.all(),
  1323. required=False
  1324. )
  1325. description = forms.CharField(
  1326. max_length=200,
  1327. required=False
  1328. )
  1329. class Meta:
  1330. nullable_fields = ['site', 'description']
  1331. class VLANGroupFilterForm(BootstrapMixin, forms.Form):
  1332. field_groups = [
  1333. ['q'],
  1334. ['region', 'sitegroup', 'site'],
  1335. ['location', 'rack']
  1336. ]
  1337. q = forms.CharField(
  1338. required=False,
  1339. widget=forms.TextInput(attrs={'placeholder': _('All Fields')}),
  1340. label=_('Search')
  1341. )
  1342. region = DynamicModelMultipleChoiceField(
  1343. queryset=Region.objects.all(),
  1344. required=False,
  1345. label=_('Region'),
  1346. fetch_trigger='open'
  1347. )
  1348. sitegroup = DynamicModelMultipleChoiceField(
  1349. queryset=SiteGroup.objects.all(),
  1350. required=False,
  1351. label=_('Site group'),
  1352. fetch_trigger='open'
  1353. )
  1354. site = DynamicModelMultipleChoiceField(
  1355. queryset=Site.objects.all(),
  1356. required=False,
  1357. label=_('Site'),
  1358. fetch_trigger='open'
  1359. )
  1360. location = DynamicModelMultipleChoiceField(
  1361. queryset=Location.objects.all(),
  1362. required=False,
  1363. label=_('Location'),
  1364. fetch_trigger='open'
  1365. )
  1366. rack = DynamicModelMultipleChoiceField(
  1367. queryset=Rack.objects.all(),
  1368. required=False,
  1369. label=_('Rack'),
  1370. fetch_trigger='open'
  1371. )
  1372. #
  1373. # VLANs
  1374. #
  1375. class VLANForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
  1376. # VLANGroup assignment fields
  1377. scope_type = forms.ChoiceField(
  1378. choices=(
  1379. ('', ''),
  1380. ('dcim.region', 'Region'),
  1381. ('dcim.sitegroup', 'Site group'),
  1382. ('dcim.site', 'Site'),
  1383. ('dcim.location', 'Location'),
  1384. ('dcim.rack', 'Rack'),
  1385. ('virtualization.clustergroup', 'Cluster group'),
  1386. ('virtualization.cluster', 'Cluster'),
  1387. ),
  1388. required=False,
  1389. widget=StaticSelect,
  1390. label='Group scope'
  1391. )
  1392. group = DynamicModelChoiceField(
  1393. queryset=VLANGroup.objects.all(),
  1394. required=False,
  1395. query_params={
  1396. 'scope_type': '$scope_type',
  1397. },
  1398. label='VLAN Group'
  1399. )
  1400. # Site assignment fields
  1401. region = DynamicModelChoiceField(
  1402. queryset=Region.objects.all(),
  1403. required=False,
  1404. initial_params={
  1405. 'sites': '$site'
  1406. },
  1407. label='Region'
  1408. )
  1409. sitegroup = DynamicModelChoiceField(
  1410. queryset=SiteGroup.objects.all(),
  1411. required=False,
  1412. initial_params={
  1413. 'sites': '$site'
  1414. },
  1415. label='Site group'
  1416. )
  1417. site = DynamicModelChoiceField(
  1418. queryset=Site.objects.all(),
  1419. required=False,
  1420. null_option='None',
  1421. query_params={
  1422. 'region_id': '$region',
  1423. 'group_id': '$sitegroup',
  1424. }
  1425. )
  1426. # Other fields
  1427. role = DynamicModelChoiceField(
  1428. queryset=Role.objects.all(),
  1429. required=False
  1430. )
  1431. tags = DynamicModelMultipleChoiceField(
  1432. queryset=Tag.objects.all(),
  1433. required=False
  1434. )
  1435. class Meta:
  1436. model = VLAN
  1437. fields = [
  1438. 'site', 'group', 'vid', 'name', 'status', 'role', 'description', 'tenant_group', 'tenant', 'tags',
  1439. ]
  1440. help_texts = {
  1441. 'site': "Leave blank if this VLAN spans multiple sites",
  1442. 'group': "VLAN group (optional)",
  1443. 'vid': "Configured VLAN ID",
  1444. 'name': "Configured VLAN name",
  1445. 'status': "Operational status of this VLAN",
  1446. 'role': "The primary function of this VLAN",
  1447. }
  1448. widgets = {
  1449. 'status': StaticSelect(),
  1450. }
  1451. class VLANCSVForm(CustomFieldModelCSVForm):
  1452. site = CSVModelChoiceField(
  1453. queryset=Site.objects.all(),
  1454. required=False,
  1455. to_field_name='name',
  1456. help_text='Assigned site'
  1457. )
  1458. group = CSVModelChoiceField(
  1459. queryset=VLANGroup.objects.all(),
  1460. required=False,
  1461. to_field_name='name',
  1462. help_text='Assigned VLAN group'
  1463. )
  1464. tenant = CSVModelChoiceField(
  1465. queryset=Tenant.objects.all(),
  1466. to_field_name='name',
  1467. required=False,
  1468. help_text='Assigned tenant'
  1469. )
  1470. status = CSVChoiceField(
  1471. choices=VLANStatusChoices,
  1472. help_text='Operational status'
  1473. )
  1474. role = CSVModelChoiceField(
  1475. queryset=Role.objects.all(),
  1476. required=False,
  1477. to_field_name='name',
  1478. help_text='Functional role'
  1479. )
  1480. class Meta:
  1481. model = VLAN
  1482. fields = ('site', 'group', 'vid', 'name', 'tenant', 'status', 'role', 'description')
  1483. help_texts = {
  1484. 'vid': 'Numeric VLAN ID (1-4095)',
  1485. 'name': 'VLAN name',
  1486. }
  1487. class VLANBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldModelBulkEditForm):
  1488. pk = forms.ModelMultipleChoiceField(
  1489. queryset=VLAN.objects.all(),
  1490. widget=forms.MultipleHiddenInput()
  1491. )
  1492. region = DynamicModelChoiceField(
  1493. queryset=Region.objects.all(),
  1494. required=False
  1495. )
  1496. site_group = DynamicModelChoiceField(
  1497. queryset=SiteGroup.objects.all(),
  1498. required=False
  1499. )
  1500. site = DynamicModelChoiceField(
  1501. queryset=Site.objects.all(),
  1502. required=False,
  1503. query_params={
  1504. 'region_id': '$region',
  1505. 'group_id': '$site_group',
  1506. }
  1507. )
  1508. group = DynamicModelChoiceField(
  1509. queryset=VLANGroup.objects.all(),
  1510. required=False,
  1511. query_params={
  1512. 'site_id': '$site'
  1513. }
  1514. )
  1515. tenant = DynamicModelChoiceField(
  1516. queryset=Tenant.objects.all(),
  1517. required=False
  1518. )
  1519. status = forms.ChoiceField(
  1520. choices=add_blank_choice(VLANStatusChoices),
  1521. required=False,
  1522. widget=StaticSelect()
  1523. )
  1524. role = DynamicModelChoiceField(
  1525. queryset=Role.objects.all(),
  1526. required=False
  1527. )
  1528. description = forms.CharField(
  1529. max_length=100,
  1530. required=False
  1531. )
  1532. class Meta:
  1533. nullable_fields = [
  1534. 'site', 'group', 'tenant', 'role', 'description',
  1535. ]
  1536. class VLANFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldModelFilterForm):
  1537. model = VLAN
  1538. field_order = [
  1539. 'q', 'region_id', 'site_group_id', 'site_id', 'group_id', 'status', 'role_id',
  1540. 'tenant_group_id', 'tenant_id',
  1541. ]
  1542. field_groups = [
  1543. ['q'],
  1544. ['region_id', 'site_group_id', 'site_id'],
  1545. ['group_id', 'role_id', 'status'],
  1546. ['tenant_group_id', 'tenant_id'],
  1547. ]
  1548. q = forms.CharField(
  1549. required=False,
  1550. widget=forms.TextInput(attrs={'placeholder': _('All Fields')}),
  1551. label=_('Search')
  1552. )
  1553. region_id = DynamicModelMultipleChoiceField(
  1554. queryset=Region.objects.all(),
  1555. required=False,
  1556. label=_('Region'),
  1557. fetch_trigger='open'
  1558. )
  1559. site_group_id = DynamicModelMultipleChoiceField(
  1560. queryset=SiteGroup.objects.all(),
  1561. required=False,
  1562. label=_('Site group'),
  1563. fetch_trigger='open'
  1564. )
  1565. site_id = DynamicModelMultipleChoiceField(
  1566. queryset=Site.objects.all(),
  1567. required=False,
  1568. null_option='None',
  1569. query_params={
  1570. 'region': '$region'
  1571. },
  1572. label=_('Site'),
  1573. fetch_trigger='open'
  1574. )
  1575. group_id = DynamicModelMultipleChoiceField(
  1576. queryset=VLANGroup.objects.all(),
  1577. required=False,
  1578. null_option='None',
  1579. query_params={
  1580. 'region': '$region'
  1581. },
  1582. label=_('VLAN group'),
  1583. fetch_trigger='open'
  1584. )
  1585. status = forms.MultipleChoiceField(
  1586. choices=VLANStatusChoices,
  1587. required=False,
  1588. widget=StaticSelectMultiple()
  1589. )
  1590. role_id = DynamicModelMultipleChoiceField(
  1591. queryset=Role.objects.all(),
  1592. required=False,
  1593. null_option='None',
  1594. label=_('Role'),
  1595. fetch_trigger='open'
  1596. )
  1597. tag = TagFilterField(model)
  1598. #
  1599. # Services
  1600. #
  1601. class ServiceForm(BootstrapMixin, CustomFieldModelForm):
  1602. ports = NumericArrayField(
  1603. base_field=forms.IntegerField(
  1604. min_value=SERVICE_PORT_MIN,
  1605. max_value=SERVICE_PORT_MAX
  1606. ),
  1607. help_text="Comma-separated list of one or more port numbers. A range may be specified using a hyphen."
  1608. )
  1609. tags = DynamicModelMultipleChoiceField(
  1610. queryset=Tag.objects.all(),
  1611. required=False
  1612. )
  1613. class Meta:
  1614. model = Service
  1615. fields = [
  1616. 'name', 'protocol', 'ports', 'ipaddresses', 'description', 'tags',
  1617. ]
  1618. help_texts = {
  1619. 'ipaddresses': "IP address assignment is optional. If no IPs are selected, the service is assumed to be "
  1620. "reachable via all IPs assigned to the device.",
  1621. }
  1622. widgets = {
  1623. 'protocol': StaticSelect(),
  1624. 'ipaddresses': StaticSelectMultiple(),
  1625. }
  1626. def __init__(self, *args, **kwargs):
  1627. super().__init__(*args, **kwargs)
  1628. # Limit IP address choices to those assigned to interfaces of the parent device/VM
  1629. if self.instance.device:
  1630. self.fields['ipaddresses'].queryset = IPAddress.objects.filter(
  1631. interface__in=self.instance.device.vc_interfaces().values_list('id', flat=True)
  1632. )
  1633. elif self.instance.virtual_machine:
  1634. self.fields['ipaddresses'].queryset = IPAddress.objects.filter(
  1635. vminterface__in=self.instance.virtual_machine.interfaces.values_list('id', flat=True)
  1636. )
  1637. else:
  1638. self.fields['ipaddresses'].choices = []
  1639. class ServiceFilterForm(BootstrapMixin, CustomFieldModelFilterForm):
  1640. model = Service
  1641. q = forms.CharField(
  1642. required=False,
  1643. widget=forms.TextInput(attrs={'placeholder': _('All Fields')}),
  1644. label=_('Search')
  1645. )
  1646. protocol = forms.ChoiceField(
  1647. choices=add_blank_choice(ServiceProtocolChoices),
  1648. required=False,
  1649. widget=StaticSelectMultiple()
  1650. )
  1651. port = forms.IntegerField(
  1652. required=False,
  1653. )
  1654. tag = TagFilterField(model)
  1655. class ServiceCSVForm(CustomFieldModelCSVForm):
  1656. device = CSVModelChoiceField(
  1657. queryset=Device.objects.all(),
  1658. required=False,
  1659. to_field_name='name',
  1660. help_text='Required if not assigned to a VM'
  1661. )
  1662. virtual_machine = CSVModelChoiceField(
  1663. queryset=VirtualMachine.objects.all(),
  1664. required=False,
  1665. to_field_name='name',
  1666. help_text='Required if not assigned to a device'
  1667. )
  1668. protocol = CSVChoiceField(
  1669. choices=ServiceProtocolChoices,
  1670. help_text='IP protocol'
  1671. )
  1672. class Meta:
  1673. model = Service
  1674. fields = ('device', 'virtual_machine', 'name', 'protocol', 'ports', 'description')
  1675. class ServiceBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldModelBulkEditForm):
  1676. pk = forms.ModelMultipleChoiceField(
  1677. queryset=Service.objects.all(),
  1678. widget=forms.MultipleHiddenInput()
  1679. )
  1680. protocol = forms.ChoiceField(
  1681. choices=add_blank_choice(ServiceProtocolChoices),
  1682. required=False,
  1683. widget=StaticSelect()
  1684. )
  1685. ports = NumericArrayField(
  1686. base_field=forms.IntegerField(
  1687. min_value=SERVICE_PORT_MIN,
  1688. max_value=SERVICE_PORT_MAX
  1689. ),
  1690. required=False
  1691. )
  1692. description = forms.CharField(
  1693. max_length=100,
  1694. required=False
  1695. )
  1696. class Meta:
  1697. nullable_fields = [
  1698. 'description',
  1699. ]