forms.py 137 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902
  1. import re
  2. from django import forms
  3. from django.contrib.auth.models import User
  4. from django.contrib.contenttypes.models import ContentType
  5. from django.contrib.postgres.forms.array import SimpleArrayField
  6. from django.core.exceptions import ObjectDoesNotExist
  7. from mptt.forms import TreeNodeChoiceField
  8. from netaddr import EUI
  9. from netaddr.core import AddrFormatError
  10. from taggit.forms import TagField
  11. from timezone_field import TimeZoneFormField
  12. from circuits.models import Circuit, Provider
  13. from extras.forms import (
  14. AddRemoveTagsForm, CustomFieldBulkEditForm, CustomFieldModelCSVForm, CustomFieldFilterForm, CustomFieldModelForm,
  15. LocalConfigContextFilterForm,
  16. )
  17. from ipam.constants import BGP_ASN_MAX, BGP_ASN_MIN
  18. from ipam.models import IPAddress, VLAN
  19. from tenancy.forms import TenancyFilterForm, TenancyForm
  20. from tenancy.models import Tenant, TenantGroup
  21. from utilities.forms import (
  22. APISelect, APISelectMultiple, add_blank_choice, ArrayFieldSelectMultiple, BootstrapMixin, BulkEditForm,
  23. BulkEditNullBooleanSelect, ColorSelect, CommentField, ConfirmationForm, CSVChoiceField, DynamicModelChoiceField,
  24. DynamicModelMultipleChoiceField, ExpandableNameField, FlexibleModelChoiceField, JSONField, SelectWithPK,
  25. SmallTextarea, SlugField, StaticSelect2, StaticSelect2Multiple, TagFilterField, BOOLEAN_WITH_BLANK_CHOICES,
  26. )
  27. from virtualization.models import Cluster, ClusterGroup, VirtualMachine
  28. from .choices import *
  29. from .constants import *
  30. from .models import (
  31. Cable, DeviceBay, DeviceBayTemplate, ConsolePort, ConsolePortTemplate, ConsoleServerPort, ConsoleServerPortTemplate,
  32. Device, DeviceRole, DeviceType, FrontPort, FrontPortTemplate, Interface, InterfaceTemplate, Manufacturer,
  33. InventoryItem, Platform, PowerFeed, PowerOutlet, PowerOutletTemplate, PowerPanel, PowerPort, PowerPortTemplate,
  34. Rack, RackGroup, RackReservation, RackRole, RearPort, RearPortTemplate, Region, Site, VirtualChassis,
  35. )
  36. DEVICE_BY_PK_RE = r'{\d+\}'
  37. INTERFACE_MODE_HELP_TEXT = """
  38. Access: One untagged VLAN<br />
  39. Tagged: One untagged VLAN and/or one or more tagged VLANs<br />
  40. Tagged (All): Implies all VLANs are available (w/optional untagged VLAN)
  41. """
  42. def get_device_by_name_or_pk(name):
  43. """
  44. Attempt to retrieve a device by either its name or primary key ('{pk}').
  45. """
  46. if re.match(DEVICE_BY_PK_RE, name):
  47. pk = name.strip('{}')
  48. device = Device.objects.get(pk=pk)
  49. else:
  50. device = Device.objects.get(name=name)
  51. return device
  52. class DeviceComponentFilterForm(BootstrapMixin, forms.Form):
  53. field_order = [
  54. 'q', 'region', 'site'
  55. ]
  56. q = forms.CharField(
  57. required=False,
  58. label='Search'
  59. )
  60. region = DynamicModelMultipleChoiceField(
  61. queryset=Region.objects.all(),
  62. to_field_name='slug',
  63. required=False,
  64. widget=APISelectMultiple(
  65. api_url='/api/dcim/regions/',
  66. value_field='slug',
  67. filter_for={
  68. 'site': 'region'
  69. }
  70. )
  71. )
  72. site = DynamicModelMultipleChoiceField(
  73. queryset=Site.objects.all(),
  74. to_field_name='slug',
  75. required=False,
  76. widget=APISelectMultiple(
  77. api_url="/api/dcim/sites/",
  78. value_field="slug",
  79. filter_for={
  80. 'device_id': 'site',
  81. }
  82. )
  83. )
  84. device_id = DynamicModelMultipleChoiceField(
  85. queryset=Device.objects.all(),
  86. required=False,
  87. label='Device',
  88. widget=APISelectMultiple(
  89. api_url='/api/dcim/devices/',
  90. )
  91. )
  92. class InterfaceCommonForm:
  93. def clean(self):
  94. super().clean()
  95. # Validate VLAN assignments
  96. tagged_vlans = self.cleaned_data['tagged_vlans']
  97. # Untagged interfaces cannot be assigned tagged VLANs
  98. if self.cleaned_data['mode'] == InterfaceModeChoices.MODE_ACCESS and tagged_vlans:
  99. raise forms.ValidationError({
  100. 'mode': "An access interface cannot have tagged VLANs assigned."
  101. })
  102. # Remove all tagged VLAN assignments from "tagged all" interfaces
  103. elif self.cleaned_data['mode'] == InterfaceModeChoices.MODE_TAGGED_ALL:
  104. self.cleaned_data['tagged_vlans'] = []
  105. # Validate tagged VLANs; must be a global VLAN or in the same site
  106. elif self.cleaned_data['mode'] == InterfaceModeChoices.MODE_TAGGED:
  107. valid_sites = [None, self.cleaned_data['device'].site]
  108. invalid_vlans = [str(v) for v in tagged_vlans if v.site not in valid_sites]
  109. if invalid_vlans:
  110. raise forms.ValidationError({
  111. 'tagged_vlans': "The tagged VLANs ({}) must belong to the same site as the interface's parent "
  112. "device/VM, or they must be global".format(', '.join(invalid_vlans))
  113. })
  114. class BulkRenameForm(forms.Form):
  115. """
  116. An extendable form to be used for renaming device components in bulk.
  117. """
  118. find = forms.CharField()
  119. replace = forms.CharField()
  120. use_regex = forms.BooleanField(
  121. required=False,
  122. initial=True,
  123. label='Use regular expressions'
  124. )
  125. def clean(self):
  126. # Validate regular expression in "find" field
  127. if self.cleaned_data['use_regex']:
  128. try:
  129. re.compile(self.cleaned_data['find'])
  130. except re.error:
  131. raise forms.ValidationError({
  132. 'find': "Invalid regular expression"
  133. })
  134. #
  135. # Fields
  136. #
  137. class MACAddressField(forms.Field):
  138. widget = forms.CharField
  139. default_error_messages = {
  140. 'invalid': 'MAC address must be in EUI-48 format',
  141. }
  142. def to_python(self, value):
  143. value = super().to_python(value)
  144. # Validate MAC address format
  145. try:
  146. value = EUI(value.strip())
  147. except AddrFormatError:
  148. raise forms.ValidationError(self.error_messages['invalid'], code='invalid')
  149. return value
  150. #
  151. # Regions
  152. #
  153. class RegionForm(BootstrapMixin, forms.ModelForm):
  154. parent = TreeNodeChoiceField(
  155. queryset=Region.objects.all(),
  156. required=False,
  157. widget=StaticSelect2()
  158. )
  159. slug = SlugField()
  160. class Meta:
  161. model = Region
  162. fields = (
  163. 'parent', 'name', 'slug',
  164. )
  165. class RegionCSVForm(forms.ModelForm):
  166. parent = forms.ModelChoiceField(
  167. queryset=Region.objects.all(),
  168. required=False,
  169. to_field_name='name',
  170. help_text='Name of parent region',
  171. error_messages={
  172. 'invalid_choice': 'Region not found.',
  173. }
  174. )
  175. class Meta:
  176. model = Region
  177. fields = Region.csv_headers
  178. help_texts = {
  179. 'name': 'Region name',
  180. 'slug': 'URL-friendly slug',
  181. }
  182. class RegionFilterForm(BootstrapMixin, forms.Form):
  183. model = Site
  184. q = forms.CharField(
  185. required=False,
  186. label='Search'
  187. )
  188. #
  189. # Sites
  190. #
  191. class SiteForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
  192. region = TreeNodeChoiceField(
  193. queryset=Region.objects.all(),
  194. required=False,
  195. widget=StaticSelect2()
  196. )
  197. slug = SlugField()
  198. comments = CommentField()
  199. tags = TagField(
  200. required=False
  201. )
  202. class Meta:
  203. model = Site
  204. fields = [
  205. 'name', 'slug', 'status', 'region', 'tenant_group', 'tenant', 'facility', 'asn', 'time_zone', 'description',
  206. 'physical_address', 'shipping_address', 'latitude', 'longitude', 'contact_name', 'contact_phone',
  207. 'contact_email', 'comments', 'tags',
  208. ]
  209. widgets = {
  210. 'physical_address': SmallTextarea(
  211. attrs={
  212. 'rows': 3,
  213. }
  214. ),
  215. 'shipping_address': SmallTextarea(
  216. attrs={
  217. 'rows': 3,
  218. }
  219. ),
  220. 'status': StaticSelect2(),
  221. 'time_zone': StaticSelect2(),
  222. }
  223. help_texts = {
  224. 'name': "Full name of the site",
  225. 'facility': "Data center provider and facility (e.g. Equinix NY7)",
  226. 'asn': "BGP autonomous system number",
  227. 'time_zone': "Local time zone",
  228. 'description': "Short description (will appear in sites list)",
  229. 'physical_address': "Physical location of the building (e.g. for GPS)",
  230. 'shipping_address': "If different from the physical address",
  231. 'latitude': "Latitude in decimal format (xx.yyyyyy)",
  232. 'longitude': "Longitude in decimal format (xx.yyyyyy)"
  233. }
  234. class SiteCSVForm(CustomFieldModelCSVForm):
  235. status = CSVChoiceField(
  236. choices=SiteStatusChoices,
  237. required=False,
  238. help_text='Operational status'
  239. )
  240. region = forms.ModelChoiceField(
  241. queryset=Region.objects.all(),
  242. required=False,
  243. to_field_name='name',
  244. help_text='Name of assigned region',
  245. error_messages={
  246. 'invalid_choice': 'Region not found.',
  247. }
  248. )
  249. tenant = forms.ModelChoiceField(
  250. queryset=Tenant.objects.all(),
  251. required=False,
  252. to_field_name='name',
  253. help_text='Name of assigned tenant',
  254. error_messages={
  255. 'invalid_choice': 'Tenant not found.',
  256. }
  257. )
  258. class Meta:
  259. model = Site
  260. fields = Site.csv_headers
  261. help_texts = {
  262. 'name': 'Site name',
  263. 'slug': 'URL-friendly slug',
  264. 'asn': '32-bit autonomous system number',
  265. }
  266. class SiteBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
  267. pk = forms.ModelMultipleChoiceField(
  268. queryset=Site.objects.all(),
  269. widget=forms.MultipleHiddenInput
  270. )
  271. status = forms.ChoiceField(
  272. choices=add_blank_choice(SiteStatusChoices),
  273. required=False,
  274. initial='',
  275. widget=StaticSelect2()
  276. )
  277. region = TreeNodeChoiceField(
  278. queryset=Region.objects.all(),
  279. required=False,
  280. widget=StaticSelect2()
  281. )
  282. tenant = DynamicModelChoiceField(
  283. queryset=Tenant.objects.all(),
  284. required=False,
  285. widget=APISelect(
  286. api_url="/api/tenancy/tenants",
  287. )
  288. )
  289. asn = forms.IntegerField(
  290. min_value=BGP_ASN_MIN,
  291. max_value=BGP_ASN_MAX,
  292. required=False,
  293. label='ASN'
  294. )
  295. description = forms.CharField(
  296. max_length=100,
  297. required=False
  298. )
  299. time_zone = TimeZoneFormField(
  300. choices=add_blank_choice(TimeZoneFormField().choices),
  301. required=False,
  302. widget=StaticSelect2()
  303. )
  304. class Meta:
  305. nullable_fields = [
  306. 'region', 'tenant', 'asn', 'description', 'time_zone',
  307. ]
  308. class SiteFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldFilterForm):
  309. model = Site
  310. field_order = ['q', 'status', 'region', 'tenant_group', 'tenant']
  311. q = forms.CharField(
  312. required=False,
  313. label='Search'
  314. )
  315. status = forms.MultipleChoiceField(
  316. choices=SiteStatusChoices,
  317. required=False,
  318. widget=StaticSelect2Multiple()
  319. )
  320. region = DynamicModelMultipleChoiceField(
  321. queryset=Region.objects.all(),
  322. to_field_name='slug',
  323. required=False,
  324. widget=APISelectMultiple(
  325. api_url="/api/dcim/regions/",
  326. value_field="slug",
  327. )
  328. )
  329. tag = TagFilterField(model)
  330. #
  331. # Rack groups
  332. #
  333. class RackGroupForm(BootstrapMixin, forms.ModelForm):
  334. site = DynamicModelChoiceField(
  335. queryset=Site.objects.all(),
  336. required=False,
  337. widget=APISelect(
  338. api_url="/api/dcim/sites/"
  339. )
  340. )
  341. slug = SlugField()
  342. class Meta:
  343. model = RackGroup
  344. fields = (
  345. 'site', 'name', 'slug',
  346. )
  347. class RackGroupCSVForm(forms.ModelForm):
  348. site = forms.ModelChoiceField(
  349. queryset=Site.objects.all(),
  350. to_field_name='name',
  351. help_text='Name of parent site',
  352. error_messages={
  353. 'invalid_choice': 'Site not found.',
  354. }
  355. )
  356. class Meta:
  357. model = RackGroup
  358. fields = RackGroup.csv_headers
  359. help_texts = {
  360. 'name': 'Name of rack group',
  361. 'slug': 'URL-friendly slug',
  362. }
  363. class RackGroupFilterForm(BootstrapMixin, forms.Form):
  364. region = DynamicModelMultipleChoiceField(
  365. queryset=Region.objects.all(),
  366. to_field_name='slug',
  367. required=False,
  368. widget=APISelectMultiple(
  369. api_url="/api/dcim/regions/",
  370. value_field="slug",
  371. filter_for={
  372. 'site': 'region'
  373. }
  374. )
  375. )
  376. site = DynamicModelMultipleChoiceField(
  377. queryset=Site.objects.all(),
  378. to_field_name='slug',
  379. required=False,
  380. widget=APISelectMultiple(
  381. api_url="/api/dcim/sites/",
  382. value_field="slug",
  383. )
  384. )
  385. #
  386. # Rack roles
  387. #
  388. class RackRoleForm(BootstrapMixin, forms.ModelForm):
  389. slug = SlugField()
  390. class Meta:
  391. model = RackRole
  392. fields = [
  393. 'name', 'slug', 'color', 'description',
  394. ]
  395. class RackRoleCSVForm(forms.ModelForm):
  396. slug = SlugField()
  397. class Meta:
  398. model = RackRole
  399. fields = RackRole.csv_headers
  400. help_texts = {
  401. 'name': 'Name of rack role',
  402. 'color': 'RGB color in hexadecimal (e.g. 00ff00)'
  403. }
  404. #
  405. # Racks
  406. #
  407. class RackForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
  408. site = DynamicModelChoiceField(
  409. queryset=Site.objects.all(),
  410. widget=APISelect(
  411. api_url="/api/dcim/sites/",
  412. filter_for={
  413. 'group': 'site_id',
  414. }
  415. )
  416. )
  417. group = DynamicModelChoiceField(
  418. queryset=RackGroup.objects.all(),
  419. required=False,
  420. widget=APISelect(
  421. api_url='/api/dcim/rack-groups/',
  422. )
  423. )
  424. role = DynamicModelChoiceField(
  425. queryset=RackRole.objects.all(),
  426. required=False,
  427. widget=APISelect(
  428. api_url='/api/dcim/rack-roles/',
  429. )
  430. )
  431. comments = CommentField()
  432. tags = TagField(
  433. required=False
  434. )
  435. class Meta:
  436. model = Rack
  437. fields = [
  438. 'site', 'group', 'name', 'facility_id', 'tenant_group', 'tenant', 'status', 'role', 'serial', 'asset_tag',
  439. 'type', 'width', 'u_height', 'desc_units', 'outer_width', 'outer_depth', 'outer_unit', 'comments', 'tags',
  440. ]
  441. help_texts = {
  442. 'site': "The site at which the rack exists",
  443. 'name': "Organizational rack name",
  444. 'facility_id': "The unique rack ID assigned by the facility",
  445. 'u_height': "Height in rack units",
  446. }
  447. widgets = {
  448. 'status': StaticSelect2(),
  449. 'type': StaticSelect2(),
  450. 'width': StaticSelect2(),
  451. 'outer_unit': StaticSelect2(),
  452. }
  453. class RackCSVForm(CustomFieldModelCSVForm):
  454. site = forms.ModelChoiceField(
  455. queryset=Site.objects.all(),
  456. to_field_name='name',
  457. help_text='Name of parent site',
  458. error_messages={
  459. 'invalid_choice': 'Site not found.',
  460. }
  461. )
  462. group_name = forms.CharField(
  463. help_text='Name of rack group',
  464. required=False
  465. )
  466. tenant = forms.ModelChoiceField(
  467. queryset=Tenant.objects.all(),
  468. required=False,
  469. to_field_name='name',
  470. help_text='Name of assigned tenant',
  471. error_messages={
  472. 'invalid_choice': 'Tenant not found.',
  473. }
  474. )
  475. status = CSVChoiceField(
  476. choices=RackStatusChoices,
  477. required=False,
  478. help_text='Operational status'
  479. )
  480. role = forms.ModelChoiceField(
  481. queryset=RackRole.objects.all(),
  482. required=False,
  483. to_field_name='name',
  484. help_text='Name of assigned role',
  485. error_messages={
  486. 'invalid_choice': 'Role not found.',
  487. }
  488. )
  489. type = CSVChoiceField(
  490. choices=RackTypeChoices,
  491. required=False,
  492. help_text='Rack type'
  493. )
  494. width = forms.ChoiceField(
  495. choices=RackWidthChoices,
  496. help_text='Rail-to-rail width (in inches)'
  497. )
  498. outer_unit = CSVChoiceField(
  499. choices=RackDimensionUnitChoices,
  500. required=False,
  501. help_text='Unit for outer dimensions'
  502. )
  503. class Meta:
  504. model = Rack
  505. fields = Rack.csv_headers
  506. help_texts = {
  507. 'name': 'Rack name',
  508. 'u_height': 'Height in rack units',
  509. }
  510. def clean(self):
  511. super().clean()
  512. site = self.cleaned_data.get('site')
  513. group_name = self.cleaned_data.get('group_name')
  514. name = self.cleaned_data.get('name')
  515. facility_id = self.cleaned_data.get('facility_id')
  516. # Validate rack group
  517. if group_name:
  518. try:
  519. self.instance.group = RackGroup.objects.get(site=site, name=group_name)
  520. except RackGroup.DoesNotExist:
  521. raise forms.ValidationError("Rack group {} not found for site {}".format(group_name, site))
  522. # Validate uniqueness of rack name within group
  523. if Rack.objects.filter(group=self.instance.group, name=name).exists():
  524. raise forms.ValidationError(
  525. "A rack named {} already exists within group {}".format(name, group_name)
  526. )
  527. # Validate uniqueness of facility ID within group
  528. if facility_id and Rack.objects.filter(group=self.instance.group, facility_id=facility_id).exists():
  529. raise forms.ValidationError(
  530. "A rack with the facility ID {} already exists within group {}".format(facility_id, group_name)
  531. )
  532. class RackBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
  533. pk = forms.ModelMultipleChoiceField(
  534. queryset=Rack.objects.all(),
  535. widget=forms.MultipleHiddenInput
  536. )
  537. site = DynamicModelChoiceField(
  538. queryset=Site.objects.all(),
  539. required=False,
  540. widget=APISelect(
  541. api_url="/api/dcim/sites",
  542. filter_for={
  543. 'group': 'site_id',
  544. }
  545. )
  546. )
  547. group = DynamicModelChoiceField(
  548. queryset=RackGroup.objects.all(),
  549. required=False,
  550. widget=APISelect(
  551. api_url="/api/dcim/rack-groups",
  552. )
  553. )
  554. tenant = DynamicModelChoiceField(
  555. queryset=Tenant.objects.all(),
  556. required=False,
  557. widget=APISelect(
  558. api_url="/api/tenancy/tenants",
  559. )
  560. )
  561. status = forms.ChoiceField(
  562. choices=add_blank_choice(RackStatusChoices),
  563. required=False,
  564. initial='',
  565. widget=StaticSelect2()
  566. )
  567. role = DynamicModelChoiceField(
  568. queryset=RackRole.objects.all(),
  569. required=False,
  570. widget=APISelect(
  571. api_url="/api/dcim/rack-roles",
  572. )
  573. )
  574. serial = forms.CharField(
  575. max_length=50,
  576. required=False,
  577. label='Serial Number'
  578. )
  579. asset_tag = forms.CharField(
  580. max_length=50,
  581. required=False
  582. )
  583. type = forms.ChoiceField(
  584. choices=add_blank_choice(RackTypeChoices),
  585. required=False,
  586. widget=StaticSelect2()
  587. )
  588. width = forms.ChoiceField(
  589. choices=add_blank_choice(RackWidthChoices),
  590. required=False,
  591. widget=StaticSelect2()
  592. )
  593. u_height = forms.IntegerField(
  594. required=False,
  595. label='Height (U)'
  596. )
  597. desc_units = forms.NullBooleanField(
  598. required=False,
  599. widget=BulkEditNullBooleanSelect,
  600. label='Descending units'
  601. )
  602. outer_width = forms.IntegerField(
  603. required=False,
  604. min_value=1
  605. )
  606. outer_depth = forms.IntegerField(
  607. required=False,
  608. min_value=1
  609. )
  610. outer_unit = forms.ChoiceField(
  611. choices=add_blank_choice(RackDimensionUnitChoices),
  612. required=False,
  613. widget=StaticSelect2()
  614. )
  615. comments = CommentField(
  616. widget=SmallTextarea,
  617. label='Comments'
  618. )
  619. class Meta:
  620. nullable_fields = [
  621. 'group', 'tenant', 'role', 'serial', 'asset_tag', 'outer_width', 'outer_depth', 'outer_unit', 'comments',
  622. ]
  623. class RackFilterForm(BootstrapMixin, TenancyFilterForm, CustomFieldFilterForm):
  624. model = Rack
  625. field_order = ['q', 'region', 'site', 'group_id', 'status', 'role', 'tenant_group', 'tenant']
  626. q = forms.CharField(
  627. required=False,
  628. label='Search'
  629. )
  630. region = DynamicModelMultipleChoiceField(
  631. queryset=Region.objects.all(),
  632. to_field_name='slug',
  633. required=False,
  634. widget=APISelectMultiple(
  635. api_url="/api/dcim/regions/",
  636. value_field="slug",
  637. filter_for={
  638. 'site': 'region'
  639. }
  640. )
  641. )
  642. site = DynamicModelMultipleChoiceField(
  643. queryset=Site.objects.all(),
  644. to_field_name='slug',
  645. required=False,
  646. widget=APISelectMultiple(
  647. api_url="/api/dcim/sites/",
  648. value_field="slug",
  649. filter_for={
  650. 'group_id': 'site'
  651. }
  652. )
  653. )
  654. group_id = DynamicModelMultipleChoiceField(
  655. queryset=RackGroup.objects.prefetch_related(
  656. 'site'
  657. ),
  658. required=False,
  659. label='Rack group',
  660. widget=APISelectMultiple(
  661. api_url="/api/dcim/rack-groups/",
  662. null_option=True
  663. )
  664. )
  665. status = forms.MultipleChoiceField(
  666. choices=RackStatusChoices,
  667. required=False,
  668. widget=StaticSelect2Multiple()
  669. )
  670. role = DynamicModelMultipleChoiceField(
  671. queryset=RackRole.objects.all(),
  672. to_field_name='slug',
  673. required=False,
  674. widget=APISelectMultiple(
  675. api_url="/api/dcim/rack-roles/",
  676. value_field="slug",
  677. null_option=True,
  678. )
  679. )
  680. tag = TagFilterField(model)
  681. #
  682. # Rack elevations
  683. #
  684. class RackElevationFilterForm(RackFilterForm):
  685. field_order = ['q', 'region', 'site', 'group_id', 'id', 'status', 'role', 'tenant_group', 'tenant']
  686. id = DynamicModelMultipleChoiceField(
  687. queryset=Rack.objects.all(),
  688. label='Rack',
  689. required=False,
  690. widget=APISelectMultiple(
  691. api_url='/api/dcim/racks/',
  692. display_field='display_name',
  693. )
  694. )
  695. def __init__(self, *args, **kwargs):
  696. super().__init__(*args, **kwargs)
  697. # Filter the rack field based on the site and group
  698. self.fields['site'].widget.add_filter_for('id', 'site')
  699. self.fields['group_id'].widget.add_filter_for('id', 'group_id')
  700. #
  701. # Rack reservations
  702. #
  703. class RackReservationForm(BootstrapMixin, TenancyForm, forms.ModelForm):
  704. units = SimpleArrayField(
  705. base_field=forms.IntegerField(),
  706. widget=ArrayFieldSelectMultiple(
  707. attrs={
  708. 'size': 10,
  709. }
  710. )
  711. )
  712. user = forms.ModelChoiceField(
  713. queryset=User.objects.order_by(
  714. 'username'
  715. ),
  716. widget=StaticSelect2()
  717. )
  718. class Meta:
  719. model = RackReservation
  720. fields = [
  721. 'units', 'user', 'tenant_group', 'tenant', 'description',
  722. ]
  723. def __init__(self, *args, **kwargs):
  724. super().__init__(*args, **kwargs)
  725. # Populate rack unit choices
  726. self.fields['units'].widget.choices = self._get_unit_choices()
  727. def _get_unit_choices(self):
  728. rack = self.instance.rack
  729. reserved_units = []
  730. for resv in rack.reservations.exclude(pk=self.instance.pk):
  731. for u in resv.units:
  732. reserved_units.append(u)
  733. unit_choices = [(u, {'label': str(u), 'disabled': u in reserved_units}) for u in rack.units]
  734. return unit_choices
  735. class RackReservationBulkEditForm(BootstrapMixin, BulkEditForm):
  736. pk = forms.ModelMultipleChoiceField(
  737. queryset=RackReservation.objects.all(),
  738. widget=forms.MultipleHiddenInput()
  739. )
  740. user = forms.ModelChoiceField(
  741. queryset=User.objects.order_by(
  742. 'username'
  743. ),
  744. required=False,
  745. widget=StaticSelect2()
  746. )
  747. tenant = DynamicModelChoiceField(
  748. queryset=Tenant.objects.all(),
  749. required=False,
  750. widget=APISelect(
  751. api_url="/api/tenancy/tenant",
  752. )
  753. )
  754. description = forms.CharField(
  755. max_length=100,
  756. required=False
  757. )
  758. class Meta:
  759. nullable_fields = []
  760. class RackReservationFilterForm(BootstrapMixin, TenancyFilterForm):
  761. field_order = ['q', 'site', 'group_id', 'tenant_group', 'tenant']
  762. q = forms.CharField(
  763. required=False,
  764. label='Search'
  765. )
  766. site = DynamicModelMultipleChoiceField(
  767. queryset=Site.objects.all(),
  768. to_field_name='slug',
  769. required=False,
  770. widget=APISelectMultiple(
  771. api_url="/api/dcim/sites/",
  772. value_field="slug",
  773. )
  774. )
  775. group_id = DynamicModelMultipleChoiceField(
  776. queryset=RackGroup.objects.prefetch_related('site'),
  777. required=False,
  778. label='Rack group',
  779. widget=APISelectMultiple(
  780. api_url="/api/dcim/rack-groups/",
  781. null_option=True,
  782. )
  783. )
  784. #
  785. # Manufacturers
  786. #
  787. class ManufacturerForm(BootstrapMixin, forms.ModelForm):
  788. slug = SlugField()
  789. class Meta:
  790. model = Manufacturer
  791. fields = [
  792. 'name', 'slug',
  793. ]
  794. class ManufacturerCSVForm(forms.ModelForm):
  795. class Meta:
  796. model = Manufacturer
  797. fields = Manufacturer.csv_headers
  798. help_texts = {
  799. 'name': 'Manufacturer name',
  800. 'slug': 'URL-friendly slug',
  801. }
  802. #
  803. # Device types
  804. #
  805. class DeviceTypeForm(BootstrapMixin, CustomFieldModelForm):
  806. manufacturer = DynamicModelChoiceField(
  807. queryset=Manufacturer.objects.all(),
  808. widget=APISelect(
  809. api_url="/api/dcim/manufacturers/",
  810. )
  811. )
  812. slug = SlugField(
  813. slug_source='model'
  814. )
  815. comments = CommentField()
  816. tags = TagField(
  817. required=False
  818. )
  819. class Meta:
  820. model = DeviceType
  821. fields = [
  822. 'manufacturer', 'model', 'slug', 'part_number', 'u_height', 'is_full_depth', 'subdevice_role', 'comments',
  823. 'tags',
  824. ]
  825. widgets = {
  826. 'subdevice_role': StaticSelect2()
  827. }
  828. class DeviceTypeImportForm(BootstrapMixin, forms.ModelForm):
  829. manufacturer = forms.ModelChoiceField(
  830. queryset=Manufacturer.objects.all(),
  831. to_field_name='name'
  832. )
  833. class Meta:
  834. model = DeviceType
  835. fields = [
  836. 'manufacturer', 'model', 'slug', 'part_number', 'u_height', 'is_full_depth', 'subdevice_role',
  837. ]
  838. class DeviceTypeBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
  839. pk = forms.ModelMultipleChoiceField(
  840. queryset=DeviceType.objects.all(),
  841. widget=forms.MultipleHiddenInput()
  842. )
  843. manufacturer = DynamicModelChoiceField(
  844. queryset=Manufacturer.objects.all(),
  845. required=False,
  846. widget=APISelect(
  847. api_url="/api/dcim/manufacturers"
  848. )
  849. )
  850. u_height = forms.IntegerField(
  851. min_value=1,
  852. required=False
  853. )
  854. is_full_depth = forms.NullBooleanField(
  855. required=False,
  856. widget=BulkEditNullBooleanSelect(),
  857. label='Is full depth'
  858. )
  859. class Meta:
  860. nullable_fields = []
  861. class DeviceTypeFilterForm(BootstrapMixin, CustomFieldFilterForm):
  862. model = DeviceType
  863. q = forms.CharField(
  864. required=False,
  865. label='Search'
  866. )
  867. manufacturer = DynamicModelMultipleChoiceField(
  868. queryset=Manufacturer.objects.all(),
  869. to_field_name='slug',
  870. required=False,
  871. widget=APISelectMultiple(
  872. api_url="/api/dcim/manufacturers/",
  873. value_field="slug",
  874. )
  875. )
  876. subdevice_role = forms.MultipleChoiceField(
  877. choices=add_blank_choice(SubdeviceRoleChoices),
  878. required=False,
  879. widget=StaticSelect2Multiple()
  880. )
  881. console_ports = forms.NullBooleanField(
  882. required=False,
  883. label='Has console ports',
  884. widget=StaticSelect2(
  885. choices=BOOLEAN_WITH_BLANK_CHOICES
  886. )
  887. )
  888. console_server_ports = forms.NullBooleanField(
  889. required=False,
  890. label='Has console server ports',
  891. widget=StaticSelect2(
  892. choices=BOOLEAN_WITH_BLANK_CHOICES
  893. )
  894. )
  895. power_ports = forms.NullBooleanField(
  896. required=False,
  897. label='Has power ports',
  898. widget=StaticSelect2(
  899. choices=BOOLEAN_WITH_BLANK_CHOICES
  900. )
  901. )
  902. power_outlets = forms.NullBooleanField(
  903. required=False,
  904. label='Has power outlets',
  905. widget=StaticSelect2(
  906. choices=BOOLEAN_WITH_BLANK_CHOICES
  907. )
  908. )
  909. interfaces = forms.NullBooleanField(
  910. required=False,
  911. label='Has interfaces',
  912. widget=StaticSelect2(
  913. choices=BOOLEAN_WITH_BLANK_CHOICES
  914. )
  915. )
  916. pass_through_ports = forms.NullBooleanField(
  917. required=False,
  918. label='Has pass-through ports',
  919. widget=StaticSelect2(
  920. choices=BOOLEAN_WITH_BLANK_CHOICES
  921. )
  922. )
  923. tag = TagFilterField(model)
  924. #
  925. # Device component templates
  926. #
  927. class ConsolePortTemplateForm(BootstrapMixin, forms.ModelForm):
  928. class Meta:
  929. model = ConsolePortTemplate
  930. fields = [
  931. 'device_type', 'name', 'type',
  932. ]
  933. widgets = {
  934. 'device_type': forms.HiddenInput(),
  935. }
  936. class ConsolePortTemplateCreateForm(BootstrapMixin, forms.Form):
  937. device_type = DynamicModelChoiceField(
  938. queryset=DeviceType.objects.all(),
  939. widget=APISelect(
  940. api_url='/api/dcim/device-types/'
  941. )
  942. )
  943. name_pattern = ExpandableNameField(
  944. label='Name'
  945. )
  946. type = forms.ChoiceField(
  947. choices=add_blank_choice(ConsolePortTypeChoices),
  948. widget=StaticSelect2()
  949. )
  950. class ConsolePortTemplateBulkEditForm(BootstrapMixin, BulkEditForm):
  951. pk = forms.ModelMultipleChoiceField(
  952. queryset=ConsolePortTemplate.objects.all(),
  953. widget=forms.MultipleHiddenInput()
  954. )
  955. type = forms.ChoiceField(
  956. choices=add_blank_choice(ConsolePortTypeChoices),
  957. required=False,
  958. widget=StaticSelect2()
  959. )
  960. class Meta:
  961. nullable_fields = ('type',)
  962. class ConsoleServerPortTemplateForm(BootstrapMixin, forms.ModelForm):
  963. class Meta:
  964. model = ConsoleServerPortTemplate
  965. fields = [
  966. 'device_type', 'name', 'type',
  967. ]
  968. widgets = {
  969. 'device_type': forms.HiddenInput(),
  970. }
  971. class ConsoleServerPortTemplateCreateForm(BootstrapMixin, forms.Form):
  972. device_type = DynamicModelChoiceField(
  973. queryset=DeviceType.objects.all(),
  974. widget=APISelect(
  975. api_url='/api/dcim/device-types/'
  976. )
  977. )
  978. name_pattern = ExpandableNameField(
  979. label='Name'
  980. )
  981. type = forms.ChoiceField(
  982. choices=add_blank_choice(ConsolePortTypeChoices),
  983. widget=StaticSelect2()
  984. )
  985. class ConsoleServerPortTemplateBulkEditForm(BootstrapMixin, BulkEditForm):
  986. pk = forms.ModelMultipleChoiceField(
  987. queryset=ConsoleServerPortTemplate.objects.all(),
  988. widget=forms.MultipleHiddenInput()
  989. )
  990. type = forms.ChoiceField(
  991. choices=add_blank_choice(ConsolePortTypeChoices),
  992. required=False,
  993. widget=StaticSelect2()
  994. )
  995. class Meta:
  996. nullable_fields = ('type',)
  997. class PowerPortTemplateForm(BootstrapMixin, forms.ModelForm):
  998. class Meta:
  999. model = PowerPortTemplate
  1000. fields = [
  1001. 'device_type', 'name', 'type', 'maximum_draw', 'allocated_draw',
  1002. ]
  1003. widgets = {
  1004. 'device_type': forms.HiddenInput(),
  1005. }
  1006. class PowerPortTemplateCreateForm(BootstrapMixin, forms.Form):
  1007. device_type = DynamicModelChoiceField(
  1008. queryset=DeviceType.objects.all(),
  1009. widget=APISelect(
  1010. api_url='/api/dcim/device-types/'
  1011. )
  1012. )
  1013. name_pattern = ExpandableNameField(
  1014. label='Name'
  1015. )
  1016. type = forms.ChoiceField(
  1017. choices=add_blank_choice(PowerPortTypeChoices),
  1018. required=False
  1019. )
  1020. maximum_draw = forms.IntegerField(
  1021. min_value=1,
  1022. required=False,
  1023. help_text="Maximum power draw (watts)"
  1024. )
  1025. allocated_draw = forms.IntegerField(
  1026. min_value=1,
  1027. required=False,
  1028. help_text="Allocated power draw (watts)"
  1029. )
  1030. class PowerPortTemplateBulkEditForm(BootstrapMixin, BulkEditForm):
  1031. pk = forms.ModelMultipleChoiceField(
  1032. queryset=PowerPortTemplate.objects.all(),
  1033. widget=forms.MultipleHiddenInput()
  1034. )
  1035. type = forms.ChoiceField(
  1036. choices=add_blank_choice(PowerPortTypeChoices),
  1037. required=False,
  1038. widget=StaticSelect2()
  1039. )
  1040. maximum_draw = forms.IntegerField(
  1041. min_value=1,
  1042. required=False,
  1043. help_text="Maximum power draw (watts)"
  1044. )
  1045. allocated_draw = forms.IntegerField(
  1046. min_value=1,
  1047. required=False,
  1048. help_text="Allocated power draw (watts)"
  1049. )
  1050. class Meta:
  1051. nullable_fields = ('type', 'maximum_draw', 'allocated_draw')
  1052. class PowerOutletTemplateForm(BootstrapMixin, forms.ModelForm):
  1053. class Meta:
  1054. model = PowerOutletTemplate
  1055. fields = [
  1056. 'device_type', 'name', 'type', 'power_port', 'feed_leg',
  1057. ]
  1058. widgets = {
  1059. 'device_type': forms.HiddenInput(),
  1060. }
  1061. def __init__(self, *args, **kwargs):
  1062. super().__init__(*args, **kwargs)
  1063. # Limit power_port choices to current DeviceType
  1064. if hasattr(self.instance, 'device_type'):
  1065. self.fields['power_port'].queryset = PowerPortTemplate.objects.filter(
  1066. device_type=self.instance.device_type
  1067. )
  1068. class PowerOutletTemplateCreateForm(BootstrapMixin, forms.Form):
  1069. device_type = DynamicModelChoiceField(
  1070. queryset=DeviceType.objects.all(),
  1071. widget=APISelect(
  1072. api_url='/api/dcim/device-types/'
  1073. )
  1074. )
  1075. name_pattern = ExpandableNameField(
  1076. label='Name'
  1077. )
  1078. type = forms.ChoiceField(
  1079. choices=add_blank_choice(PowerOutletTypeChoices),
  1080. required=False
  1081. )
  1082. power_port = forms.ModelChoiceField(
  1083. queryset=PowerPortTemplate.objects.all(),
  1084. required=False
  1085. )
  1086. feed_leg = forms.ChoiceField(
  1087. choices=add_blank_choice(PowerOutletFeedLegChoices),
  1088. required=False,
  1089. widget=StaticSelect2()
  1090. )
  1091. def __init__(self, *args, **kwargs):
  1092. super().__init__(*args, **kwargs)
  1093. # Limit power_port choices to current DeviceType
  1094. device_type = DeviceType.objects.get(
  1095. pk=self.initial.get('device_type') or self.data.get('device_type')
  1096. )
  1097. self.fields['power_port'].queryset = PowerPortTemplate.objects.filter(
  1098. device_type=device_type
  1099. )
  1100. class PowerOutletTemplateBulkEditForm(BootstrapMixin, BulkEditForm):
  1101. pk = forms.ModelMultipleChoiceField(
  1102. queryset=PowerOutletTemplate.objects.all(),
  1103. widget=forms.MultipleHiddenInput()
  1104. )
  1105. type = forms.ChoiceField(
  1106. choices=add_blank_choice(PowerOutletTypeChoices),
  1107. required=False,
  1108. widget=StaticSelect2()
  1109. )
  1110. feed_leg = forms.ChoiceField(
  1111. choices=add_blank_choice(PowerOutletFeedLegChoices),
  1112. required=False,
  1113. widget=StaticSelect2()
  1114. )
  1115. class Meta:
  1116. nullable_fields = ('type', 'feed_leg')
  1117. class InterfaceTemplateForm(BootstrapMixin, forms.ModelForm):
  1118. class Meta:
  1119. model = InterfaceTemplate
  1120. fields = [
  1121. 'device_type', 'name', 'type', 'mgmt_only',
  1122. ]
  1123. widgets = {
  1124. 'device_type': forms.HiddenInput(),
  1125. 'type': StaticSelect2(),
  1126. }
  1127. class InterfaceTemplateCreateForm(BootstrapMixin, forms.Form):
  1128. device_type = DynamicModelChoiceField(
  1129. queryset=DeviceType.objects.all(),
  1130. widget=APISelect(
  1131. api_url='/api/dcim/device-types/'
  1132. )
  1133. )
  1134. name_pattern = ExpandableNameField(
  1135. label='Name'
  1136. )
  1137. type = forms.ChoiceField(
  1138. choices=InterfaceTypeChoices,
  1139. widget=StaticSelect2()
  1140. )
  1141. mgmt_only = forms.BooleanField(
  1142. required=False,
  1143. label='Management only'
  1144. )
  1145. class InterfaceTemplateBulkEditForm(BootstrapMixin, BulkEditForm):
  1146. pk = forms.ModelMultipleChoiceField(
  1147. queryset=InterfaceTemplate.objects.all(),
  1148. widget=forms.MultipleHiddenInput()
  1149. )
  1150. type = forms.ChoiceField(
  1151. choices=add_blank_choice(InterfaceTypeChoices),
  1152. required=False,
  1153. widget=StaticSelect2()
  1154. )
  1155. mgmt_only = forms.NullBooleanField(
  1156. required=False,
  1157. widget=BulkEditNullBooleanSelect,
  1158. label='Management only'
  1159. )
  1160. class Meta:
  1161. nullable_fields = []
  1162. class FrontPortTemplateForm(BootstrapMixin, forms.ModelForm):
  1163. class Meta:
  1164. model = FrontPortTemplate
  1165. fields = [
  1166. 'device_type', 'name', 'type', 'rear_port', 'rear_port_position',
  1167. ]
  1168. widgets = {
  1169. 'device_type': forms.HiddenInput(),
  1170. 'rear_port': StaticSelect2(),
  1171. }
  1172. def __init__(self, *args, **kwargs):
  1173. super().__init__(*args, **kwargs)
  1174. # Limit rear_port choices to current DeviceType
  1175. if hasattr(self.instance, 'device_type'):
  1176. self.fields['rear_port'].queryset = RearPortTemplate.objects.filter(
  1177. device_type=self.instance.device_type
  1178. )
  1179. class FrontPortTemplateCreateForm(BootstrapMixin, forms.Form):
  1180. device_type = DynamicModelChoiceField(
  1181. queryset=DeviceType.objects.all(),
  1182. widget=APISelect(
  1183. api_url='/api/dcim/device-types/'
  1184. )
  1185. )
  1186. name_pattern = ExpandableNameField(
  1187. label='Name'
  1188. )
  1189. type = forms.ChoiceField(
  1190. choices=PortTypeChoices,
  1191. widget=StaticSelect2()
  1192. )
  1193. rear_port_set = forms.MultipleChoiceField(
  1194. choices=[],
  1195. label='Rear ports',
  1196. help_text='Select one rear port assignment for each front port being created.',
  1197. )
  1198. def __init__(self, *args, **kwargs):
  1199. super().__init__(*args, **kwargs)
  1200. device_type = DeviceType.objects.get(
  1201. pk=self.initial.get('device_type') or self.data.get('device_type')
  1202. )
  1203. # Determine which rear port positions are occupied. These will be excluded from the list of available mappings.
  1204. occupied_port_positions = [
  1205. (front_port.rear_port_id, front_port.rear_port_position)
  1206. for front_port in device_type.frontport_templates.all()
  1207. ]
  1208. # Populate rear port choices
  1209. choices = []
  1210. rear_ports = RearPortTemplate.objects.filter(device_type=device_type)
  1211. for rear_port in rear_ports:
  1212. for i in range(1, rear_port.positions + 1):
  1213. if (rear_port.pk, i) not in occupied_port_positions:
  1214. choices.append(
  1215. ('{}:{}'.format(rear_port.pk, i), '{}:{}'.format(rear_port.name, i))
  1216. )
  1217. self.fields['rear_port_set'].choices = choices
  1218. def clean(self):
  1219. # Validate that the number of ports being created equals the number of selected (rear port, position) tuples
  1220. front_port_count = len(self.cleaned_data['name_pattern'])
  1221. rear_port_count = len(self.cleaned_data['rear_port_set'])
  1222. if front_port_count != rear_port_count:
  1223. raise forms.ValidationError({
  1224. 'rear_port_set': 'The provided name pattern will create {} ports, however {} rear port assignments '
  1225. 'were selected. These counts must match.'.format(front_port_count, rear_port_count)
  1226. })
  1227. def get_iterative_data(self, iteration):
  1228. # Assign rear port and position from selected set
  1229. rear_port, position = self.cleaned_data['rear_port_set'][iteration].split(':')
  1230. return {
  1231. 'rear_port': int(rear_port),
  1232. 'rear_port_position': int(position),
  1233. }
  1234. class FrontPortTemplateBulkEditForm(BootstrapMixin, BulkEditForm):
  1235. pk = forms.ModelMultipleChoiceField(
  1236. queryset=FrontPortTemplate.objects.all(),
  1237. widget=forms.MultipleHiddenInput()
  1238. )
  1239. type = forms.ChoiceField(
  1240. choices=add_blank_choice(PortTypeChoices),
  1241. required=False,
  1242. widget=StaticSelect2()
  1243. )
  1244. class Meta:
  1245. nullable_fields = ()
  1246. class RearPortTemplateForm(BootstrapMixin, forms.ModelForm):
  1247. class Meta:
  1248. model = RearPortTemplate
  1249. fields = [
  1250. 'device_type', 'name', 'type', 'positions',
  1251. ]
  1252. widgets = {
  1253. 'device_type': forms.HiddenInput(),
  1254. 'type': StaticSelect2(),
  1255. }
  1256. class RearPortTemplateCreateForm(BootstrapMixin, forms.Form):
  1257. device_type = DynamicModelChoiceField(
  1258. queryset=DeviceType.objects.all(),
  1259. widget=APISelect(
  1260. api_url='/api/dcim/device-types/'
  1261. )
  1262. )
  1263. name_pattern = ExpandableNameField(
  1264. label='Name'
  1265. )
  1266. type = forms.ChoiceField(
  1267. choices=PortTypeChoices,
  1268. widget=StaticSelect2(),
  1269. )
  1270. positions = forms.IntegerField(
  1271. min_value=REARPORT_POSITIONS_MIN,
  1272. max_value=REARPORT_POSITIONS_MAX,
  1273. initial=1,
  1274. help_text='The number of front ports which may be mapped to each rear port'
  1275. )
  1276. class RearPortTemplateBulkEditForm(BootstrapMixin, BulkEditForm):
  1277. pk = forms.ModelMultipleChoiceField(
  1278. queryset=RearPortTemplate.objects.all(),
  1279. widget=forms.MultipleHiddenInput()
  1280. )
  1281. type = forms.ChoiceField(
  1282. choices=add_blank_choice(PortTypeChoices),
  1283. required=False,
  1284. widget=StaticSelect2()
  1285. )
  1286. class Meta:
  1287. nullable_fields = ()
  1288. class DeviceBayTemplateForm(BootstrapMixin, forms.ModelForm):
  1289. class Meta:
  1290. model = DeviceBayTemplate
  1291. fields = [
  1292. 'device_type', 'name',
  1293. ]
  1294. widgets = {
  1295. 'device_type': forms.HiddenInput(),
  1296. }
  1297. class DeviceBayTemplateCreateForm(BootstrapMixin, forms.Form):
  1298. device_type = DynamicModelChoiceField(
  1299. queryset=DeviceType.objects.all(),
  1300. widget=APISelect(
  1301. api_url='/api/dcim/device-types/'
  1302. )
  1303. )
  1304. name_pattern = ExpandableNameField(
  1305. label='Name'
  1306. )
  1307. # TODO: DeviceBayTemplate has no fields suitable for bulk-editing yet
  1308. # class DeviceBayTemplateBulkEditForm(BootstrapMixin, BulkEditForm):
  1309. # pk = forms.ModelMultipleChoiceField(
  1310. # queryset=FrontPortTemplate.objects.all(),
  1311. # widget=forms.MultipleHiddenInput()
  1312. # )
  1313. #
  1314. # class Meta:
  1315. # nullable_fields = ()
  1316. #
  1317. # Component template import forms
  1318. #
  1319. class ComponentTemplateImportForm(BootstrapMixin, forms.ModelForm):
  1320. def __init__(self, device_type, data=None, *args, **kwargs):
  1321. # Must pass the parent DeviceType on form initialization
  1322. data.update({
  1323. 'device_type': device_type.pk,
  1324. })
  1325. super().__init__(data, *args, **kwargs)
  1326. def clean_device_type(self):
  1327. data = self.cleaned_data['device_type']
  1328. # Limit fields referencing other components to the parent DeviceType
  1329. for field_name, field in self.fields.items():
  1330. if isinstance(field, forms.ModelChoiceField) and field_name != 'device_type':
  1331. field.queryset = field.queryset.filter(device_type=data)
  1332. return data
  1333. class ConsolePortTemplateImportForm(ComponentTemplateImportForm):
  1334. class Meta:
  1335. model = ConsolePortTemplate
  1336. fields = [
  1337. 'device_type', 'name', 'type',
  1338. ]
  1339. class ConsoleServerPortTemplateImportForm(ComponentTemplateImportForm):
  1340. class Meta:
  1341. model = ConsoleServerPortTemplate
  1342. fields = [
  1343. 'device_type', 'name', 'type',
  1344. ]
  1345. class PowerPortTemplateImportForm(ComponentTemplateImportForm):
  1346. class Meta:
  1347. model = PowerPortTemplate
  1348. fields = [
  1349. 'device_type', 'name', 'type', 'maximum_draw', 'allocated_draw',
  1350. ]
  1351. class PowerOutletTemplateImportForm(ComponentTemplateImportForm):
  1352. power_port = forms.ModelChoiceField(
  1353. queryset=PowerPortTemplate.objects.all(),
  1354. to_field_name='name',
  1355. required=False
  1356. )
  1357. class Meta:
  1358. model = PowerOutletTemplate
  1359. fields = [
  1360. 'device_type', 'name', 'type', 'power_port', 'feed_leg',
  1361. ]
  1362. class InterfaceTemplateImportForm(ComponentTemplateImportForm):
  1363. type = forms.ChoiceField(
  1364. choices=InterfaceTypeChoices.CHOICES
  1365. )
  1366. class Meta:
  1367. model = InterfaceTemplate
  1368. fields = [
  1369. 'device_type', 'name', 'type', 'mgmt_only',
  1370. ]
  1371. class FrontPortTemplateImportForm(ComponentTemplateImportForm):
  1372. type = forms.ChoiceField(
  1373. choices=PortTypeChoices.CHOICES
  1374. )
  1375. rear_port = forms.ModelChoiceField(
  1376. queryset=RearPortTemplate.objects.all(),
  1377. to_field_name='name',
  1378. required=False
  1379. )
  1380. class Meta:
  1381. model = FrontPortTemplate
  1382. fields = [
  1383. 'device_type', 'name', 'type', 'rear_port', 'rear_port_position',
  1384. ]
  1385. class RearPortTemplateImportForm(ComponentTemplateImportForm):
  1386. type = forms.ChoiceField(
  1387. choices=PortTypeChoices.CHOICES
  1388. )
  1389. class Meta:
  1390. model = RearPortTemplate
  1391. fields = [
  1392. 'device_type', 'name', 'type', 'positions',
  1393. ]
  1394. class DeviceBayTemplateImportForm(ComponentTemplateImportForm):
  1395. class Meta:
  1396. model = DeviceBayTemplate
  1397. fields = [
  1398. 'device_type', 'name',
  1399. ]
  1400. #
  1401. # Device roles
  1402. #
  1403. class DeviceRoleForm(BootstrapMixin, forms.ModelForm):
  1404. slug = SlugField()
  1405. class Meta:
  1406. model = DeviceRole
  1407. fields = [
  1408. 'name', 'slug', 'color', 'vm_role', 'description',
  1409. ]
  1410. class DeviceRoleCSVForm(forms.ModelForm):
  1411. slug = SlugField()
  1412. class Meta:
  1413. model = DeviceRole
  1414. fields = DeviceRole.csv_headers
  1415. help_texts = {
  1416. 'name': 'Name of device role',
  1417. 'color': 'RGB color in hexadecimal (e.g. 00ff00)'
  1418. }
  1419. #
  1420. # Platforms
  1421. #
  1422. class PlatformForm(BootstrapMixin, forms.ModelForm):
  1423. manufacturer = DynamicModelChoiceField(
  1424. queryset=Manufacturer.objects.all(),
  1425. required=False,
  1426. widget=APISelect(
  1427. api_url="/api/dcim/manufacturers/",
  1428. )
  1429. )
  1430. slug = SlugField(
  1431. max_length=64
  1432. )
  1433. class Meta:
  1434. model = Platform
  1435. fields = [
  1436. 'name', 'slug', 'manufacturer', 'napalm_driver', 'napalm_args',
  1437. ]
  1438. widgets = {
  1439. 'napalm_args': SmallTextarea(),
  1440. }
  1441. class PlatformCSVForm(forms.ModelForm):
  1442. slug = SlugField()
  1443. manufacturer = forms.ModelChoiceField(
  1444. queryset=Manufacturer.objects.all(),
  1445. required=False,
  1446. to_field_name='name',
  1447. help_text='Manufacturer name',
  1448. error_messages={
  1449. 'invalid_choice': 'Manufacturer not found.',
  1450. }
  1451. )
  1452. class Meta:
  1453. model = Platform
  1454. fields = Platform.csv_headers
  1455. help_texts = {
  1456. 'name': 'Platform name',
  1457. }
  1458. #
  1459. # Devices
  1460. #
  1461. class DeviceForm(BootstrapMixin, TenancyForm, CustomFieldModelForm):
  1462. site = DynamicModelChoiceField(
  1463. queryset=Site.objects.all(),
  1464. widget=APISelect(
  1465. api_url="/api/dcim/sites/",
  1466. filter_for={
  1467. 'rack': 'site_id'
  1468. }
  1469. )
  1470. )
  1471. rack = DynamicModelChoiceField(
  1472. queryset=Rack.objects.all(),
  1473. required=False,
  1474. widget=APISelect(
  1475. api_url='/api/dcim/racks/',
  1476. display_field='display_name'
  1477. )
  1478. )
  1479. position = forms.TypedChoiceField(
  1480. required=False,
  1481. empty_value=None,
  1482. help_text="The lowest-numbered unit occupied by the device",
  1483. widget=APISelect(
  1484. api_url='/api/dcim/racks/{{rack}}/elevation/',
  1485. disabled_indicator='device'
  1486. )
  1487. )
  1488. manufacturer = DynamicModelChoiceField(
  1489. queryset=Manufacturer.objects.all(),
  1490. required=False,
  1491. widget=APISelect(
  1492. api_url="/api/dcim/manufacturers/",
  1493. filter_for={
  1494. 'device_type': 'manufacturer_id',
  1495. 'platform': 'manufacturer_id'
  1496. }
  1497. )
  1498. )
  1499. device_type = DynamicModelChoiceField(
  1500. queryset=DeviceType.objects.all(),
  1501. widget=APISelect(
  1502. api_url='/api/dcim/device-types/',
  1503. display_field='model'
  1504. )
  1505. )
  1506. device_role = DynamicModelChoiceField(
  1507. queryset=DeviceRole.objects.all(),
  1508. widget=APISelect(
  1509. api_url='/api/dcim/device-roles/'
  1510. )
  1511. )
  1512. platform = DynamicModelChoiceField(
  1513. queryset=Platform.objects.all(),
  1514. required=False,
  1515. widget=APISelect(
  1516. api_url="/api/dcim/platforms/",
  1517. additional_query_params={
  1518. "manufacturer_id": "null"
  1519. }
  1520. )
  1521. )
  1522. cluster_group = DynamicModelChoiceField(
  1523. queryset=ClusterGroup.objects.all(),
  1524. required=False,
  1525. widget=APISelect(
  1526. api_url="/api/virtualization/cluster-groups/",
  1527. filter_for={
  1528. 'cluster': 'group_id'
  1529. },
  1530. attrs={
  1531. 'nullable': 'true'
  1532. }
  1533. )
  1534. )
  1535. cluster = DynamicModelChoiceField(
  1536. queryset=Cluster.objects.all(),
  1537. required=False,
  1538. widget=APISelect(
  1539. api_url='/api/virtualization/clusters/',
  1540. )
  1541. )
  1542. comments = CommentField()
  1543. tags = TagField(required=False)
  1544. local_context_data = JSONField(
  1545. required=False,
  1546. label=''
  1547. )
  1548. class Meta:
  1549. model = Device
  1550. fields = [
  1551. 'name', 'device_role', 'device_type', 'serial', 'asset_tag', 'site', 'rack', 'position', 'face',
  1552. 'status', 'platform', 'primary_ip4', 'primary_ip6', 'cluster_group', 'cluster', 'tenant_group', 'tenant',
  1553. 'comments', 'tags', 'local_context_data'
  1554. ]
  1555. help_texts = {
  1556. 'device_role': "The function this device serves",
  1557. 'serial': "Chassis serial number",
  1558. 'local_context_data': "Local config context data overwrites all source contexts in the final rendered "
  1559. "config context",
  1560. }
  1561. widgets = {
  1562. 'face': StaticSelect2(
  1563. filter_for={
  1564. 'position': 'face'
  1565. }
  1566. ),
  1567. 'status': StaticSelect2(),
  1568. 'primary_ip4': StaticSelect2(),
  1569. 'primary_ip6': StaticSelect2(),
  1570. }
  1571. def __init__(self, *args, **kwargs):
  1572. # Initialize helper selectors
  1573. instance = kwargs.get('instance')
  1574. if 'initial' not in kwargs:
  1575. kwargs['initial'] = {}
  1576. # Using hasattr() instead of "is not None" to avoid RelatedObjectDoesNotExist on required field
  1577. if instance and hasattr(instance, 'device_type'):
  1578. kwargs['initial']['manufacturer'] = instance.device_type.manufacturer
  1579. if instance and instance.cluster is not None:
  1580. kwargs['initial']['cluster_group'] = instance.cluster.group
  1581. if 'device_type' in kwargs['initial'] and 'manufacturer' not in kwargs['initial']:
  1582. device_type_id = kwargs['initial']['device_type']
  1583. manufacturer_id = DeviceType.objects.filter(pk=device_type_id).values_list('manufacturer__pk', flat=True).first()
  1584. kwargs['initial']['manufacturer'] = manufacturer_id
  1585. if 'cluster' in kwargs['initial'] and 'cluster_group' not in kwargs['initial']:
  1586. cluster_id = kwargs['initial']['cluster']
  1587. cluster_group_id = Cluster.objects.filter(pk=cluster_id).values_list('group__pk', flat=True).first()
  1588. kwargs['initial']['cluster_group'] = cluster_group_id
  1589. super().__init__(*args, **kwargs)
  1590. if self.instance.pk:
  1591. # Compile list of choices for primary IPv4 and IPv6 addresses
  1592. for family in [4, 6]:
  1593. ip_choices = [(None, '---------')]
  1594. # Gather PKs of all interfaces belonging to this Device or a peer VirtualChassis member
  1595. interface_ids = self.instance.vc_interfaces.values('pk')
  1596. # Collect interface IPs
  1597. interface_ips = IPAddress.objects.prefetch_related('interface').filter(
  1598. family=family, interface_id__in=interface_ids
  1599. )
  1600. if interface_ips:
  1601. ip_list = [(ip.id, '{} ({})'.format(ip.address, ip.interface)) for ip in interface_ips]
  1602. ip_choices.append(('Interface IPs', ip_list))
  1603. # Collect NAT IPs
  1604. nat_ips = IPAddress.objects.prefetch_related('nat_inside').filter(
  1605. family=family, nat_inside__interface__in=interface_ids
  1606. )
  1607. if nat_ips:
  1608. ip_list = [(ip.id, '{} ({})'.format(ip.address, ip.nat_inside.address)) for ip in nat_ips]
  1609. ip_choices.append(('NAT IPs', ip_list))
  1610. self.fields['primary_ip{}'.format(family)].choices = ip_choices
  1611. # If editing an existing device, exclude it from the list of occupied rack units. This ensures that a device
  1612. # can be flipped from one face to another.
  1613. self.fields['position'].widget.add_additional_query_param('exclude', self.instance.pk)
  1614. # Limit platform by manufacturer
  1615. self.fields['platform'].queryset = Platform.objects.filter(
  1616. Q(manufacturer__isnull=True) | Q(manufacturer=self.instance.device_type.manufacturer)
  1617. )
  1618. else:
  1619. # An object that doesn't exist yet can't have any IPs assigned to it
  1620. self.fields['primary_ip4'].choices = []
  1621. self.fields['primary_ip4'].widget.attrs['readonly'] = True
  1622. self.fields['primary_ip6'].choices = []
  1623. self.fields['primary_ip6'].widget.attrs['readonly'] = True
  1624. # Rack position
  1625. pk = self.instance.pk if self.instance.pk else None
  1626. try:
  1627. if self.is_bound and self.data.get('rack') and str(self.data.get('face')):
  1628. position_choices = Rack.objects.get(pk=self.data['rack']) \
  1629. .get_rack_units(face=self.data.get('face'), exclude=pk)
  1630. elif self.initial.get('rack') and str(self.initial.get('face')):
  1631. position_choices = Rack.objects.get(pk=self.initial['rack']) \
  1632. .get_rack_units(face=self.initial.get('face'), exclude=pk)
  1633. else:
  1634. position_choices = []
  1635. except Rack.DoesNotExist:
  1636. position_choices = []
  1637. self.fields['position'].choices = [('', '---------')] + [
  1638. (p['id'], {
  1639. 'label': p['name'],
  1640. 'disabled': bool(p['device'] and p['id'] != self.initial.get('position')),
  1641. }) for p in position_choices
  1642. ]
  1643. # Disable rack assignment if this is a child device installed in a parent device
  1644. if pk and self.instance.device_type.is_child_device and hasattr(self.instance, 'parent_bay'):
  1645. self.fields['site'].disabled = True
  1646. self.fields['rack'].disabled = True
  1647. self.initial['site'] = self.instance.parent_bay.device.site_id
  1648. self.initial['rack'] = self.instance.parent_bay.device.rack_id
  1649. class BaseDeviceCSVForm(CustomFieldModelCSVForm):
  1650. device_role = forms.ModelChoiceField(
  1651. queryset=DeviceRole.objects.all(),
  1652. to_field_name='name',
  1653. help_text='Name of assigned role',
  1654. error_messages={
  1655. 'invalid_choice': 'Invalid device role.',
  1656. }
  1657. )
  1658. tenant = forms.ModelChoiceField(
  1659. queryset=Tenant.objects.all(),
  1660. required=False,
  1661. to_field_name='name',
  1662. help_text='Name of assigned tenant',
  1663. error_messages={
  1664. 'invalid_choice': 'Tenant not found.',
  1665. }
  1666. )
  1667. manufacturer = forms.ModelChoiceField(
  1668. queryset=Manufacturer.objects.all(),
  1669. to_field_name='name',
  1670. help_text='Device type manufacturer',
  1671. error_messages={
  1672. 'invalid_choice': 'Invalid manufacturer.',
  1673. }
  1674. )
  1675. model_name = forms.CharField(
  1676. help_text='Device type model name'
  1677. )
  1678. platform = forms.ModelChoiceField(
  1679. queryset=Platform.objects.all(),
  1680. required=False,
  1681. to_field_name='name',
  1682. help_text='Name of assigned platform',
  1683. error_messages={
  1684. 'invalid_choice': 'Invalid platform.',
  1685. }
  1686. )
  1687. status = CSVChoiceField(
  1688. choices=DeviceStatusChoices,
  1689. help_text='Operational status'
  1690. )
  1691. class Meta:
  1692. fields = []
  1693. model = Device
  1694. help_texts = {
  1695. 'name': 'Device name',
  1696. }
  1697. def clean(self):
  1698. super().clean()
  1699. manufacturer = self.cleaned_data.get('manufacturer')
  1700. model_name = self.cleaned_data.get('model_name')
  1701. # Validate device type
  1702. if manufacturer and model_name:
  1703. try:
  1704. self.instance.device_type = DeviceType.objects.get(manufacturer=manufacturer, model=model_name)
  1705. except DeviceType.DoesNotExist:
  1706. raise forms.ValidationError("Device type {} {} not found".format(manufacturer, model_name))
  1707. class DeviceCSVForm(BaseDeviceCSVForm):
  1708. site = forms.ModelChoiceField(
  1709. queryset=Site.objects.all(),
  1710. to_field_name='name',
  1711. help_text='Name of parent site',
  1712. error_messages={
  1713. 'invalid_choice': 'Invalid site name.',
  1714. }
  1715. )
  1716. rack_group = forms.CharField(
  1717. required=False,
  1718. help_text='Parent rack\'s group (if any)'
  1719. )
  1720. rack_name = forms.CharField(
  1721. required=False,
  1722. help_text='Name of parent rack'
  1723. )
  1724. face = CSVChoiceField(
  1725. choices=DeviceFaceChoices,
  1726. required=False,
  1727. help_text='Mounted rack face'
  1728. )
  1729. cluster = forms.ModelChoiceField(
  1730. queryset=Cluster.objects.all(),
  1731. to_field_name='name',
  1732. required=False,
  1733. help_text='Virtualization cluster',
  1734. error_messages={
  1735. 'invalid_choice': 'Invalid cluster name.',
  1736. }
  1737. )
  1738. class Meta(BaseDeviceCSVForm.Meta):
  1739. fields = [
  1740. 'name', 'device_role', 'tenant', 'manufacturer', 'model_name', 'platform', 'serial', 'asset_tag', 'status',
  1741. 'site', 'rack_group', 'rack_name', 'position', 'face', 'cluster', 'comments',
  1742. ]
  1743. def clean(self):
  1744. super().clean()
  1745. site = self.cleaned_data.get('site')
  1746. rack_group = self.cleaned_data.get('rack_group')
  1747. rack_name = self.cleaned_data.get('rack_name')
  1748. # Validate rack
  1749. if site and rack_group and rack_name:
  1750. try:
  1751. self.instance.rack = Rack.objects.get(site=site, group__name=rack_group, name=rack_name)
  1752. except Rack.DoesNotExist:
  1753. raise forms.ValidationError("Rack {} not found in site {} group {}".format(rack_name, site, rack_group))
  1754. elif site and rack_name:
  1755. try:
  1756. self.instance.rack = Rack.objects.get(site=site, group__isnull=True, name=rack_name)
  1757. except Rack.DoesNotExist:
  1758. raise forms.ValidationError("Rack {} not found in site {} (no group)".format(rack_name, site))
  1759. class ChildDeviceCSVForm(BaseDeviceCSVForm):
  1760. parent = FlexibleModelChoiceField(
  1761. queryset=Device.objects.all(),
  1762. to_field_name='name',
  1763. help_text='Name or ID of parent device',
  1764. error_messages={
  1765. 'invalid_choice': 'Parent device not found.',
  1766. }
  1767. )
  1768. device_bay_name = forms.CharField(
  1769. help_text='Name of device bay',
  1770. )
  1771. cluster = forms.ModelChoiceField(
  1772. queryset=Cluster.objects.all(),
  1773. to_field_name='name',
  1774. required=False,
  1775. help_text='Virtualization cluster',
  1776. error_messages={
  1777. 'invalid_choice': 'Invalid cluster name.',
  1778. }
  1779. )
  1780. class Meta(BaseDeviceCSVForm.Meta):
  1781. fields = [
  1782. 'name', 'device_role', 'tenant', 'manufacturer', 'model_name', 'platform', 'serial', 'asset_tag', 'status',
  1783. 'parent', 'device_bay_name', 'cluster', 'comments',
  1784. ]
  1785. def clean(self):
  1786. super().clean()
  1787. parent = self.cleaned_data.get('parent')
  1788. device_bay_name = self.cleaned_data.get('device_bay_name')
  1789. # Validate device bay
  1790. if parent and device_bay_name:
  1791. try:
  1792. self.instance.parent_bay = DeviceBay.objects.get(device=parent, name=device_bay_name)
  1793. # Inherit site and rack from parent device
  1794. self.instance.site = parent.site
  1795. self.instance.rack = parent.rack
  1796. except DeviceBay.DoesNotExist:
  1797. raise forms.ValidationError("Parent device/bay ({} {}) not found".format(parent, device_bay_name))
  1798. class DeviceBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
  1799. pk = forms.ModelMultipleChoiceField(
  1800. queryset=Device.objects.all(),
  1801. widget=forms.MultipleHiddenInput()
  1802. )
  1803. device_type = DynamicModelChoiceField(
  1804. queryset=DeviceType.objects.all(),
  1805. required=False,
  1806. widget=APISelect(
  1807. api_url="/api/dcim/device-types/",
  1808. display_field='display_name'
  1809. )
  1810. )
  1811. device_role = DynamicModelChoiceField(
  1812. queryset=DeviceRole.objects.all(),
  1813. required=False,
  1814. widget=APISelect(
  1815. api_url="/api/dcim/device-roles/"
  1816. )
  1817. )
  1818. tenant = DynamicModelChoiceField(
  1819. queryset=Tenant.objects.all(),
  1820. required=False,
  1821. widget=APISelect(
  1822. api_url="/api/tenancy/tenants/"
  1823. )
  1824. )
  1825. platform = DynamicModelChoiceField(
  1826. queryset=Platform.objects.all(),
  1827. required=False,
  1828. widget=APISelect(
  1829. api_url="/api/dcim/platforms/"
  1830. )
  1831. )
  1832. status = forms.ChoiceField(
  1833. choices=add_blank_choice(DeviceStatusChoices),
  1834. required=False,
  1835. widget=StaticSelect2()
  1836. )
  1837. serial = forms.CharField(
  1838. max_length=50,
  1839. required=False,
  1840. label='Serial Number'
  1841. )
  1842. class Meta:
  1843. nullable_fields = [
  1844. 'tenant', 'platform', 'serial',
  1845. ]
  1846. class DeviceFilterForm(BootstrapMixin, LocalConfigContextFilterForm, TenancyFilterForm, CustomFieldFilterForm):
  1847. model = Device
  1848. field_order = [
  1849. 'q', 'region', 'site', 'rack_group_id', 'rack_id', 'status', 'role', 'tenant_group', 'tenant',
  1850. 'manufacturer_id', 'device_type_id', 'mac_address', 'has_primary_ip',
  1851. ]
  1852. q = forms.CharField(
  1853. required=False,
  1854. label='Search'
  1855. )
  1856. region = DynamicModelMultipleChoiceField(
  1857. queryset=Region.objects.all(),
  1858. to_field_name='slug',
  1859. required=False,
  1860. widget=APISelectMultiple(
  1861. api_url="/api/dcim/regions/",
  1862. value_field="slug",
  1863. filter_for={
  1864. 'site': 'region'
  1865. }
  1866. )
  1867. )
  1868. site = DynamicModelMultipleChoiceField(
  1869. queryset=Site.objects.all(),
  1870. to_field_name='slug',
  1871. required=False,
  1872. widget=APISelectMultiple(
  1873. api_url="/api/dcim/sites/",
  1874. value_field="slug",
  1875. filter_for={
  1876. 'rack_group_id': 'site',
  1877. 'rack_id': 'site',
  1878. }
  1879. )
  1880. )
  1881. rack_group_id = DynamicModelMultipleChoiceField(
  1882. queryset=RackGroup.objects.all(),
  1883. required=False,
  1884. label='Rack group',
  1885. widget=APISelectMultiple(
  1886. api_url="/api/dcim/rack-groups/",
  1887. filter_for={
  1888. 'rack_id': 'group_id',
  1889. }
  1890. )
  1891. )
  1892. rack_id = DynamicModelMultipleChoiceField(
  1893. queryset=Rack.objects.all(),
  1894. required=False,
  1895. label='Rack',
  1896. widget=APISelectMultiple(
  1897. api_url="/api/dcim/racks/",
  1898. null_option=True,
  1899. )
  1900. )
  1901. role = DynamicModelMultipleChoiceField(
  1902. queryset=DeviceRole.objects.all(),
  1903. to_field_name='slug',
  1904. required=False,
  1905. widget=APISelectMultiple(
  1906. api_url="/api/dcim/device-roles/",
  1907. value_field="slug",
  1908. )
  1909. )
  1910. manufacturer_id = DynamicModelMultipleChoiceField(
  1911. queryset=Manufacturer.objects.all(),
  1912. required=False,
  1913. label='Manufacturer',
  1914. widget=APISelectMultiple(
  1915. api_url="/api/dcim/manufacturers/",
  1916. filter_for={
  1917. 'device_type_id': 'manufacturer_id',
  1918. }
  1919. )
  1920. )
  1921. device_type_id = DynamicModelMultipleChoiceField(
  1922. queryset=DeviceType.objects.all(),
  1923. required=False,
  1924. label='Model',
  1925. widget=APISelectMultiple(
  1926. api_url="/api/dcim/device-types/",
  1927. display_field="model",
  1928. )
  1929. )
  1930. platform = DynamicModelMultipleChoiceField(
  1931. queryset=Platform.objects.all(),
  1932. to_field_name='slug',
  1933. required=False,
  1934. widget=APISelectMultiple(
  1935. api_url="/api/dcim/platforms/",
  1936. value_field="slug",
  1937. null_option=True,
  1938. )
  1939. )
  1940. status = forms.MultipleChoiceField(
  1941. choices=DeviceStatusChoices,
  1942. required=False,
  1943. widget=StaticSelect2Multiple()
  1944. )
  1945. mac_address = forms.CharField(
  1946. required=False,
  1947. label='MAC address'
  1948. )
  1949. has_primary_ip = forms.NullBooleanField(
  1950. required=False,
  1951. label='Has a primary IP',
  1952. widget=StaticSelect2(
  1953. choices=BOOLEAN_WITH_BLANK_CHOICES
  1954. )
  1955. )
  1956. virtual_chassis_member = forms.NullBooleanField(
  1957. required=False,
  1958. label='Virtual chassis member',
  1959. widget=StaticSelect2(
  1960. choices=BOOLEAN_WITH_BLANK_CHOICES
  1961. )
  1962. )
  1963. console_ports = forms.NullBooleanField(
  1964. required=False,
  1965. label='Has console ports',
  1966. widget=StaticSelect2(
  1967. choices=BOOLEAN_WITH_BLANK_CHOICES
  1968. )
  1969. )
  1970. console_server_ports = forms.NullBooleanField(
  1971. required=False,
  1972. label='Has console server ports',
  1973. widget=StaticSelect2(
  1974. choices=BOOLEAN_WITH_BLANK_CHOICES
  1975. )
  1976. )
  1977. power_ports = forms.NullBooleanField(
  1978. required=False,
  1979. label='Has power ports',
  1980. widget=StaticSelect2(
  1981. choices=BOOLEAN_WITH_BLANK_CHOICES
  1982. )
  1983. )
  1984. power_outlets = forms.NullBooleanField(
  1985. required=False,
  1986. label='Has power outlets',
  1987. widget=StaticSelect2(
  1988. choices=BOOLEAN_WITH_BLANK_CHOICES
  1989. )
  1990. )
  1991. interfaces = forms.NullBooleanField(
  1992. required=False,
  1993. label='Has interfaces',
  1994. widget=StaticSelect2(
  1995. choices=BOOLEAN_WITH_BLANK_CHOICES
  1996. )
  1997. )
  1998. pass_through_ports = forms.NullBooleanField(
  1999. required=False,
  2000. label='Has pass-through ports',
  2001. widget=StaticSelect2(
  2002. choices=BOOLEAN_WITH_BLANK_CHOICES
  2003. )
  2004. )
  2005. tag = TagFilterField(model)
  2006. #
  2007. # Bulk device component creation
  2008. #
  2009. class DeviceBulkAddComponentForm(BootstrapMixin, forms.Form):
  2010. pk = forms.ModelMultipleChoiceField(
  2011. queryset=Device.objects.all(),
  2012. widget=forms.MultipleHiddenInput()
  2013. )
  2014. name_pattern = ExpandableNameField(
  2015. label='Name'
  2016. )
  2017. class DeviceBulkAddInterfaceForm(DeviceBulkAddComponentForm):
  2018. type = forms.ChoiceField(
  2019. choices=InterfaceTypeChoices,
  2020. widget=StaticSelect2()
  2021. )
  2022. enabled = forms.BooleanField(
  2023. required=False,
  2024. initial=True
  2025. )
  2026. mtu = forms.IntegerField(
  2027. required=False,
  2028. min_value=INTERFACE_MTU_MIN,
  2029. max_value=INTERFACE_MTU_MAX,
  2030. label='MTU'
  2031. )
  2032. mgmt_only = forms.BooleanField(
  2033. required=False,
  2034. label='Management only'
  2035. )
  2036. description = forms.CharField(
  2037. max_length=100,
  2038. required=False
  2039. )
  2040. #
  2041. # Console ports
  2042. #
  2043. class ConsolePortFilterForm(DeviceComponentFilterForm):
  2044. model = ConsolePort
  2045. tag = TagFilterField(model)
  2046. class ConsolePortForm(BootstrapMixin, forms.ModelForm):
  2047. tags = TagField(
  2048. required=False
  2049. )
  2050. class Meta:
  2051. model = ConsolePort
  2052. fields = [
  2053. 'device', 'name', 'type', 'description', 'tags',
  2054. ]
  2055. widgets = {
  2056. 'device': forms.HiddenInput(),
  2057. }
  2058. class ConsolePortCreateForm(BootstrapMixin, forms.Form):
  2059. device = DynamicModelChoiceField(
  2060. queryset=Device.objects.prefetch_related('device_type__manufacturer'),
  2061. widget=APISelect(
  2062. api_url="/api/dcim/devices/",
  2063. )
  2064. )
  2065. name_pattern = ExpandableNameField(
  2066. label='Name'
  2067. )
  2068. type = forms.ChoiceField(
  2069. choices=add_blank_choice(ConsolePortTypeChoices),
  2070. required=False,
  2071. widget=StaticSelect2()
  2072. )
  2073. description = forms.CharField(
  2074. max_length=100,
  2075. required=False
  2076. )
  2077. tags = TagField(
  2078. required=False
  2079. )
  2080. class ConsolePortBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
  2081. pk = forms.ModelMultipleChoiceField(
  2082. queryset=ConsolePort.objects.all(),
  2083. widget=forms.MultipleHiddenInput()
  2084. )
  2085. type = forms.ChoiceField(
  2086. choices=add_blank_choice(ConsolePortTypeChoices),
  2087. required=False,
  2088. widget=StaticSelect2()
  2089. )
  2090. description = forms.CharField(
  2091. max_length=100,
  2092. required=False
  2093. )
  2094. class Meta:
  2095. nullable_fields = (
  2096. 'description',
  2097. )
  2098. class ConsolePortCSVForm(forms.ModelForm):
  2099. device = FlexibleModelChoiceField(
  2100. queryset=Device.objects.all(),
  2101. to_field_name='name',
  2102. help_text='Name or ID of device',
  2103. error_messages={
  2104. 'invalid_choice': 'Device not found.',
  2105. }
  2106. )
  2107. class Meta:
  2108. model = ConsolePort
  2109. fields = ConsolePort.csv_headers
  2110. #
  2111. # Console server ports
  2112. #
  2113. class ConsoleServerPortFilterForm(DeviceComponentFilterForm):
  2114. model = ConsoleServerPort
  2115. tag = TagFilterField(model)
  2116. class ConsoleServerPortForm(BootstrapMixin, forms.ModelForm):
  2117. tags = TagField(
  2118. required=False
  2119. )
  2120. class Meta:
  2121. model = ConsoleServerPort
  2122. fields = [
  2123. 'device', 'name', 'type', 'description', 'tags',
  2124. ]
  2125. widgets = {
  2126. 'device': forms.HiddenInput(),
  2127. }
  2128. class ConsoleServerPortCreateForm(BootstrapMixin, forms.Form):
  2129. device = DynamicModelChoiceField(
  2130. queryset=Device.objects.prefetch_related('device_type__manufacturer'),
  2131. widget=APISelect(
  2132. api_url="/api/dcim/devices/",
  2133. )
  2134. )
  2135. name_pattern = ExpandableNameField(
  2136. label='Name'
  2137. )
  2138. type = forms.ChoiceField(
  2139. choices=add_blank_choice(ConsolePortTypeChoices),
  2140. required=False,
  2141. widget=StaticSelect2()
  2142. )
  2143. description = forms.CharField(
  2144. max_length=100,
  2145. required=False
  2146. )
  2147. tags = TagField(
  2148. required=False
  2149. )
  2150. class ConsoleServerPortBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
  2151. pk = forms.ModelMultipleChoiceField(
  2152. queryset=ConsoleServerPort.objects.all(),
  2153. widget=forms.MultipleHiddenInput()
  2154. )
  2155. type = forms.ChoiceField(
  2156. choices=add_blank_choice(ConsolePortTypeChoices),
  2157. required=False,
  2158. widget=StaticSelect2()
  2159. )
  2160. description = forms.CharField(
  2161. max_length=100,
  2162. required=False
  2163. )
  2164. class Meta:
  2165. nullable_fields = [
  2166. 'description',
  2167. ]
  2168. class ConsoleServerPortBulkRenameForm(BulkRenameForm):
  2169. pk = forms.ModelMultipleChoiceField(
  2170. queryset=ConsoleServerPort.objects.all(),
  2171. widget=forms.MultipleHiddenInput()
  2172. )
  2173. class ConsoleServerPortBulkDisconnectForm(ConfirmationForm):
  2174. pk = forms.ModelMultipleChoiceField(
  2175. queryset=ConsoleServerPort.objects.all(),
  2176. widget=forms.MultipleHiddenInput()
  2177. )
  2178. class ConsoleServerPortCSVForm(forms.ModelForm):
  2179. device = FlexibleModelChoiceField(
  2180. queryset=Device.objects.all(),
  2181. to_field_name='name',
  2182. help_text='Name or ID of device',
  2183. error_messages={
  2184. 'invalid_choice': 'Device not found.',
  2185. }
  2186. )
  2187. class Meta:
  2188. model = ConsoleServerPort
  2189. fields = ConsoleServerPort.csv_headers
  2190. #
  2191. # Power ports
  2192. #
  2193. class PowerPortFilterForm(DeviceComponentFilterForm):
  2194. model = PowerPort
  2195. tag = TagFilterField(model)
  2196. class PowerPortForm(BootstrapMixin, forms.ModelForm):
  2197. tags = TagField(
  2198. required=False
  2199. )
  2200. class Meta:
  2201. model = PowerPort
  2202. fields = [
  2203. 'device', 'name', 'type', 'maximum_draw', 'allocated_draw', 'description', 'tags',
  2204. ]
  2205. widgets = {
  2206. 'device': forms.HiddenInput(),
  2207. }
  2208. class PowerPortCreateForm(BootstrapMixin, forms.Form):
  2209. device = DynamicModelChoiceField(
  2210. queryset=Device.objects.prefetch_related('device_type__manufacturer'),
  2211. widget=APISelect(
  2212. api_url="/api/dcim/devices/",
  2213. )
  2214. )
  2215. name_pattern = ExpandableNameField(
  2216. label='Name'
  2217. )
  2218. type = forms.ChoiceField(
  2219. choices=add_blank_choice(PowerPortTypeChoices),
  2220. required=False,
  2221. widget=StaticSelect2()
  2222. )
  2223. maximum_draw = forms.IntegerField(
  2224. min_value=1,
  2225. required=False,
  2226. help_text="Maximum draw in watts"
  2227. )
  2228. allocated_draw = forms.IntegerField(
  2229. min_value=1,
  2230. required=False,
  2231. help_text="Allocated draw in watts"
  2232. )
  2233. description = forms.CharField(
  2234. max_length=100,
  2235. required=False
  2236. )
  2237. tags = TagField(
  2238. required=False
  2239. )
  2240. class PowerPortBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
  2241. pk = forms.ModelMultipleChoiceField(
  2242. queryset=PowerPort.objects.all(),
  2243. widget=forms.MultipleHiddenInput()
  2244. )
  2245. type = forms.ChoiceField(
  2246. choices=add_blank_choice(PowerPortTypeChoices),
  2247. required=False,
  2248. widget=StaticSelect2()
  2249. )
  2250. maximum_draw = forms.IntegerField(
  2251. min_value=1,
  2252. required=False,
  2253. help_text="Maximum draw in watts"
  2254. )
  2255. allocated_draw = forms.IntegerField(
  2256. min_value=1,
  2257. required=False,
  2258. help_text="Allocated draw in watts"
  2259. )
  2260. description = forms.CharField(
  2261. max_length=100,
  2262. required=False
  2263. )
  2264. class Meta:
  2265. nullable_fields = (
  2266. 'description',
  2267. )
  2268. class PowerPortCSVForm(forms.ModelForm):
  2269. device = FlexibleModelChoiceField(
  2270. queryset=Device.objects.all(),
  2271. to_field_name='name',
  2272. help_text='Name or ID of device',
  2273. error_messages={
  2274. 'invalid_choice': 'Device not found.',
  2275. }
  2276. )
  2277. class Meta:
  2278. model = PowerPort
  2279. fields = PowerPort.csv_headers
  2280. #
  2281. # Power outlets
  2282. #
  2283. class PowerOutletFilterForm(DeviceComponentFilterForm):
  2284. model = PowerOutlet
  2285. tag = TagFilterField(model)
  2286. class PowerOutletForm(BootstrapMixin, forms.ModelForm):
  2287. power_port = forms.ModelChoiceField(
  2288. queryset=PowerPort.objects.all(),
  2289. required=False
  2290. )
  2291. tags = TagField(
  2292. required=False
  2293. )
  2294. class Meta:
  2295. model = PowerOutlet
  2296. fields = [
  2297. 'device', 'name', 'type', 'power_port', 'feed_leg', 'description', 'tags',
  2298. ]
  2299. widgets = {
  2300. 'device': forms.HiddenInput(),
  2301. }
  2302. def __init__(self, *args, **kwargs):
  2303. super().__init__(*args, **kwargs)
  2304. # Limit power_port choices to the local device
  2305. if hasattr(self.instance, 'device'):
  2306. self.fields['power_port'].queryset = PowerPort.objects.filter(
  2307. device=self.instance.device
  2308. )
  2309. class PowerOutletCreateForm(BootstrapMixin, forms.Form):
  2310. device = DynamicModelChoiceField(
  2311. queryset=Device.objects.prefetch_related('device_type__manufacturer'),
  2312. widget=APISelect(
  2313. api_url="/api/dcim/devices/",
  2314. )
  2315. )
  2316. name_pattern = ExpandableNameField(
  2317. label='Name'
  2318. )
  2319. type = forms.ChoiceField(
  2320. choices=add_blank_choice(PowerOutletTypeChoices),
  2321. required=False,
  2322. widget=StaticSelect2()
  2323. )
  2324. power_port = forms.ModelChoiceField(
  2325. queryset=PowerPort.objects.all(),
  2326. required=False
  2327. )
  2328. feed_leg = forms.ChoiceField(
  2329. choices=add_blank_choice(PowerOutletFeedLegChoices),
  2330. required=False
  2331. )
  2332. description = forms.CharField(
  2333. max_length=100,
  2334. required=False
  2335. )
  2336. tags = TagField(
  2337. required=False
  2338. )
  2339. def __init__(self, *args, **kwargs):
  2340. super().__init__(*args, **kwargs)
  2341. # Limit power_port queryset to PowerPorts which belong to the parent Device
  2342. device = Device.objects.get(
  2343. pk=self.initial.get('device') or self.data.get('device')
  2344. )
  2345. self.fields['power_port'].queryset = PowerPort.objects.filter(device=device)
  2346. class PowerOutletCSVForm(forms.ModelForm):
  2347. device = FlexibleModelChoiceField(
  2348. queryset=Device.objects.all(),
  2349. to_field_name='name',
  2350. help_text='Name or ID of device',
  2351. error_messages={
  2352. 'invalid_choice': 'Device not found.',
  2353. }
  2354. )
  2355. power_port = FlexibleModelChoiceField(
  2356. queryset=PowerPort.objects.all(),
  2357. required=False,
  2358. to_field_name='name',
  2359. help_text='Name or ID of Power Port',
  2360. error_messages={
  2361. 'invalid_choice': 'Power Port not found.',
  2362. }
  2363. )
  2364. feed_leg = CSVChoiceField(
  2365. choices=PowerOutletFeedLegChoices,
  2366. required=False,
  2367. )
  2368. class Meta:
  2369. model = PowerOutlet
  2370. fields = PowerOutlet.csv_headers
  2371. def __init__(self, *args, **kwargs):
  2372. super().__init__(*args, **kwargs)
  2373. # Limit PowerPort choices to those belonging to this device (or VC master)
  2374. if self.is_bound:
  2375. try:
  2376. device = self.fields['device'].to_python(self.data['device'])
  2377. except forms.ValidationError:
  2378. device = None
  2379. else:
  2380. try:
  2381. device = self.instance.device
  2382. except Device.DoesNotExist:
  2383. device = None
  2384. if device:
  2385. self.fields['power_port'].queryset = PowerPort.objects.filter(
  2386. device__in=[device, device.get_vc_master()]
  2387. )
  2388. else:
  2389. self.fields['power_port'].queryset = PowerPort.objects.none()
  2390. class PowerOutletBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
  2391. pk = forms.ModelMultipleChoiceField(
  2392. queryset=PowerOutlet.objects.all(),
  2393. widget=forms.MultipleHiddenInput()
  2394. )
  2395. device = forms.ModelChoiceField(
  2396. queryset=Device.objects.all(),
  2397. required=False,
  2398. widget=forms.HiddenInput()
  2399. )
  2400. type = forms.ChoiceField(
  2401. choices=add_blank_choice(PowerOutletTypeChoices),
  2402. required=False
  2403. )
  2404. feed_leg = forms.ChoiceField(
  2405. choices=add_blank_choice(PowerOutletFeedLegChoices),
  2406. required=False,
  2407. )
  2408. power_port = forms.ModelChoiceField(
  2409. queryset=PowerPort.objects.all(),
  2410. required=False
  2411. )
  2412. description = forms.CharField(
  2413. max_length=100,
  2414. required=False
  2415. )
  2416. class Meta:
  2417. nullable_fields = [
  2418. 'type', 'feed_leg', 'power_port', 'description',
  2419. ]
  2420. def __init__(self, *args, **kwargs):
  2421. super().__init__(*args, **kwargs)
  2422. # Limit power_port queryset to PowerPorts which belong to the parent Device
  2423. if 'device' in self.initial:
  2424. device = Device.objects.filter(pk=self.initial['device']).first()
  2425. self.fields['power_port'].queryset = PowerPort.objects.filter(device=device)
  2426. else:
  2427. self.fields['power_port'].choices = ()
  2428. self.fields['power_port'].widget.attrs['disabled'] = True
  2429. class PowerOutletBulkRenameForm(BulkRenameForm):
  2430. pk = forms.ModelMultipleChoiceField(
  2431. queryset=PowerOutlet.objects.all(),
  2432. widget=forms.MultipleHiddenInput
  2433. )
  2434. class PowerOutletBulkDisconnectForm(ConfirmationForm):
  2435. pk = forms.ModelMultipleChoiceField(
  2436. queryset=PowerOutlet.objects.all(),
  2437. widget=forms.MultipleHiddenInput
  2438. )
  2439. #
  2440. # Interfaces
  2441. #
  2442. class InterfaceFilterForm(DeviceComponentFilterForm):
  2443. model = Interface
  2444. tag = TagFilterField(model)
  2445. class InterfaceForm(InterfaceCommonForm, BootstrapMixin, forms.ModelForm):
  2446. untagged_vlan = DynamicModelChoiceField(
  2447. queryset=VLAN.objects.all(),
  2448. required=False,
  2449. label='Untagged VLAN',
  2450. widget=APISelect(
  2451. api_url="/api/ipam/vlans/",
  2452. display_field='display_name',
  2453. full=True,
  2454. additional_query_params={
  2455. 'site_id': 'null',
  2456. },
  2457. )
  2458. )
  2459. tagged_vlans = DynamicModelMultipleChoiceField(
  2460. queryset=VLAN.objects.all(),
  2461. required=False,
  2462. label='Tagged VLANs',
  2463. widget=APISelectMultiple(
  2464. api_url="/api/ipam/vlans/",
  2465. display_field='display_name',
  2466. full=True,
  2467. additional_query_params={
  2468. 'site_id': 'null',
  2469. },
  2470. )
  2471. )
  2472. tags = TagField(
  2473. required=False
  2474. )
  2475. class Meta:
  2476. model = Interface
  2477. fields = [
  2478. 'device', 'name', 'type', 'enabled', 'lag', 'mac_address', 'mtu', 'mgmt_only', 'description',
  2479. 'mode', 'untagged_vlan', 'tagged_vlans', 'tags',
  2480. ]
  2481. widgets = {
  2482. 'device': forms.HiddenInput(),
  2483. 'type': StaticSelect2(),
  2484. 'lag': StaticSelect2(),
  2485. 'mode': StaticSelect2(),
  2486. }
  2487. labels = {
  2488. 'mode': '802.1Q Mode',
  2489. }
  2490. help_texts = {
  2491. 'mode': INTERFACE_MODE_HELP_TEXT,
  2492. }
  2493. def __init__(self, *args, **kwargs):
  2494. super().__init__(*args, **kwargs)
  2495. if self.is_bound:
  2496. device = Device.objects.get(pk=self.data['device'])
  2497. else:
  2498. device = self.instance.device
  2499. # Limit LAG choices to interfaces belonging to this device (or VC master)
  2500. self.fields['lag'].queryset = Interface.objects.filter(
  2501. device__in=[device, device.get_vc_master()],
  2502. type=InterfaceTypeChoices.TYPE_LAG
  2503. )
  2504. # Add current site to VLANs query params
  2505. self.fields['untagged_vlan'].widget.add_additional_query_param('site_id', device.site.pk)
  2506. self.fields['tagged_vlans'].widget.add_additional_query_param('site_id', device.site.pk)
  2507. class InterfaceCreateForm(BootstrapMixin, InterfaceCommonForm, forms.Form):
  2508. device = DynamicModelChoiceField(
  2509. queryset=Device.objects.prefetch_related('device_type__manufacturer'),
  2510. widget=APISelect(
  2511. api_url="/api/dcim/devices/",
  2512. )
  2513. )
  2514. name_pattern = ExpandableNameField(
  2515. label='Name'
  2516. )
  2517. type = forms.ChoiceField(
  2518. choices=InterfaceTypeChoices,
  2519. widget=StaticSelect2(),
  2520. )
  2521. enabled = forms.BooleanField(
  2522. required=False,
  2523. initial=True
  2524. )
  2525. lag = forms.ModelChoiceField(
  2526. queryset=Interface.objects.all(),
  2527. required=False,
  2528. label='Parent LAG',
  2529. widget=StaticSelect2(),
  2530. )
  2531. mtu = forms.IntegerField(
  2532. required=False,
  2533. min_value=INTERFACE_MTU_MIN,
  2534. max_value=INTERFACE_MTU_MAX,
  2535. label='MTU'
  2536. )
  2537. mac_address = forms.CharField(
  2538. required=False,
  2539. label='MAC Address'
  2540. )
  2541. mgmt_only = forms.BooleanField(
  2542. required=False,
  2543. label='Management only',
  2544. help_text='This interface is used only for out-of-band management'
  2545. )
  2546. description = forms.CharField(
  2547. max_length=100,
  2548. required=False
  2549. )
  2550. mode = forms.ChoiceField(
  2551. choices=add_blank_choice(InterfaceModeChoices),
  2552. required=False,
  2553. widget=StaticSelect2(),
  2554. )
  2555. tags = TagField(
  2556. required=False
  2557. )
  2558. untagged_vlan = DynamicModelChoiceField(
  2559. queryset=VLAN.objects.all(),
  2560. required=False,
  2561. widget=APISelect(
  2562. api_url="/api/ipam/vlans/",
  2563. display_field='display_name',
  2564. full=True,
  2565. additional_query_params={
  2566. 'site_id': 'null',
  2567. },
  2568. )
  2569. )
  2570. tagged_vlans = DynamicModelMultipleChoiceField(
  2571. queryset=VLAN.objects.all(),
  2572. required=False,
  2573. widget=APISelectMultiple(
  2574. api_url="/api/ipam/vlans/",
  2575. display_field='display_name',
  2576. full=True,
  2577. additional_query_params={
  2578. 'site_id': 'null',
  2579. },
  2580. )
  2581. )
  2582. def __init__(self, *args, **kwargs):
  2583. super().__init__(*args, **kwargs)
  2584. # Limit LAG choices to interfaces which belong to the parent device (or VC master)
  2585. device = Device.objects.get(
  2586. pk=self.initial.get('device') or self.data.get('device')
  2587. )
  2588. self.fields['lag'].queryset = Interface.objects.filter(
  2589. device__in=[device, device.get_vc_master()],
  2590. type=InterfaceTypeChoices.TYPE_LAG
  2591. )
  2592. # Add current site to VLANs query params
  2593. self.fields['untagged_vlan'].widget.add_additional_query_param('site_id', device.site.pk)
  2594. self.fields['tagged_vlans'].widget.add_additional_query_param('site_id', device.site.pk)
  2595. class InterfaceCSVForm(forms.ModelForm):
  2596. device = FlexibleModelChoiceField(
  2597. queryset=Device.objects.all(),
  2598. required=False,
  2599. to_field_name='name',
  2600. help_text='Name or ID of device',
  2601. error_messages={
  2602. 'invalid_choice': 'Device not found.',
  2603. }
  2604. )
  2605. virtual_machine = FlexibleModelChoiceField(
  2606. queryset=VirtualMachine.objects.all(),
  2607. required=False,
  2608. to_field_name='name',
  2609. help_text='Name or ID of virtual machine',
  2610. error_messages={
  2611. 'invalid_choice': 'Virtual machine not found.',
  2612. }
  2613. )
  2614. lag = FlexibleModelChoiceField(
  2615. queryset=Interface.objects.all(),
  2616. required=False,
  2617. to_field_name='name',
  2618. help_text='Name or ID of LAG interface',
  2619. error_messages={
  2620. 'invalid_choice': 'LAG interface not found.',
  2621. }
  2622. )
  2623. type = CSVChoiceField(
  2624. choices=InterfaceTypeChoices,
  2625. )
  2626. mode = CSVChoiceField(
  2627. choices=InterfaceModeChoices,
  2628. required=False,
  2629. )
  2630. class Meta:
  2631. model = Interface
  2632. fields = Interface.csv_headers
  2633. def __init__(self, *args, **kwargs):
  2634. super().__init__(*args, **kwargs)
  2635. # Limit LAG choices to interfaces belonging to this device (or VC master)
  2636. if self.is_bound and 'device' in self.data:
  2637. try:
  2638. device = self.fields['device'].to_python(self.data['device'])
  2639. except forms.ValidationError:
  2640. device = None
  2641. else:
  2642. device = self.instance.device
  2643. if device:
  2644. self.fields['lag'].queryset = Interface.objects.filter(
  2645. device__in=[device, device.get_vc_master()], type=InterfaceTypeChoices.TYPE_LAG
  2646. )
  2647. else:
  2648. self.fields['lag'].queryset = Interface.objects.none()
  2649. def clean_enabled(self):
  2650. # Make sure enabled is True when it's not included in the uploaded data
  2651. if 'enabled' not in self.data:
  2652. return True
  2653. else:
  2654. return self.cleaned_data['enabled']
  2655. class InterfaceBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
  2656. pk = forms.ModelMultipleChoiceField(
  2657. queryset=Interface.objects.all(),
  2658. widget=forms.MultipleHiddenInput()
  2659. )
  2660. device = forms.ModelChoiceField(
  2661. queryset=Device.objects.all(),
  2662. required=False,
  2663. widget=forms.HiddenInput()
  2664. )
  2665. type = forms.ChoiceField(
  2666. choices=add_blank_choice(InterfaceTypeChoices),
  2667. required=False,
  2668. widget=StaticSelect2()
  2669. )
  2670. enabled = forms.NullBooleanField(
  2671. required=False,
  2672. widget=BulkEditNullBooleanSelect()
  2673. )
  2674. lag = forms.ModelChoiceField(
  2675. queryset=Interface.objects.all(),
  2676. required=False,
  2677. label='Parent LAG',
  2678. widget=StaticSelect2()
  2679. )
  2680. mac_address = forms.CharField(
  2681. required=False,
  2682. label='MAC Address'
  2683. )
  2684. mtu = forms.IntegerField(
  2685. required=False,
  2686. min_value=INTERFACE_MTU_MIN,
  2687. max_value=INTERFACE_MTU_MAX,
  2688. label='MTU'
  2689. )
  2690. mgmt_only = forms.NullBooleanField(
  2691. required=False,
  2692. widget=BulkEditNullBooleanSelect(),
  2693. label='Management only'
  2694. )
  2695. description = forms.CharField(
  2696. max_length=100,
  2697. required=False
  2698. )
  2699. mode = forms.ChoiceField(
  2700. choices=add_blank_choice(InterfaceModeChoices),
  2701. required=False,
  2702. widget=StaticSelect2()
  2703. )
  2704. untagged_vlan = DynamicModelChoiceField(
  2705. queryset=VLAN.objects.all(),
  2706. required=False,
  2707. widget=APISelect(
  2708. api_url="/api/ipam/vlans/",
  2709. display_field='display_name',
  2710. full=True,
  2711. additional_query_params={
  2712. 'site_id': 'null',
  2713. },
  2714. )
  2715. )
  2716. tagged_vlans = DynamicModelMultipleChoiceField(
  2717. queryset=VLAN.objects.all(),
  2718. required=False,
  2719. widget=APISelectMultiple(
  2720. api_url="/api/ipam/vlans/",
  2721. display_field='display_name',
  2722. full=True,
  2723. additional_query_params={
  2724. 'site_id': 'null',
  2725. },
  2726. )
  2727. )
  2728. class Meta:
  2729. nullable_fields = [
  2730. 'lag', 'mac_address', 'mtu', 'description', 'mode', 'untagged_vlan', 'tagged_vlans'
  2731. ]
  2732. def __init__(self, *args, **kwargs):
  2733. super().__init__(*args, **kwargs)
  2734. # Limit LAG choices to interfaces which belong to the parent device (or VC master)
  2735. if 'device' in self.initial:
  2736. device = Device.objects.filter(pk=self.initial['device']).first()
  2737. self.fields['lag'].queryset = Interface.objects.filter(
  2738. device__in=[device, device.get_vc_master()],
  2739. type=InterfaceTypeChoices.TYPE_LAG
  2740. )
  2741. # Add current site to VLANs query params
  2742. self.fields['untagged_vlan'].widget.add_additional_query_param('site_id', device.site.pk)
  2743. self.fields['tagged_vlans'].widget.add_additional_query_param('site_id', device.site.pk)
  2744. else:
  2745. self.fields['lag'].choices = ()
  2746. self.fields['lag'].widget.attrs['disabled'] = True
  2747. def clean(self):
  2748. # Untagged interfaces cannot be assigned tagged VLANs
  2749. if self.cleaned_data['mode'] == InterfaceModeChoices.MODE_ACCESS and self.cleaned_data['tagged_vlans']:
  2750. raise forms.ValidationError({
  2751. 'mode': "An access interface cannot have tagged VLANs assigned."
  2752. })
  2753. # Remove all tagged VLAN assignments from "tagged all" interfaces
  2754. elif self.cleaned_data['mode'] == InterfaceModeChoices.MODE_TAGGED_ALL:
  2755. self.cleaned_data['tagged_vlans'] = []
  2756. class InterfaceBulkRenameForm(BulkRenameForm):
  2757. pk = forms.ModelMultipleChoiceField(
  2758. queryset=Interface.objects.all(),
  2759. widget=forms.MultipleHiddenInput()
  2760. )
  2761. class InterfaceBulkDisconnectForm(ConfirmationForm):
  2762. pk = forms.ModelMultipleChoiceField(
  2763. queryset=Interface.objects.all(),
  2764. widget=forms.MultipleHiddenInput()
  2765. )
  2766. #
  2767. # Front pass-through ports
  2768. #
  2769. class FrontPortFilterForm(DeviceComponentFilterForm):
  2770. model = FrontPort
  2771. tag = TagFilterField(model)
  2772. class FrontPortForm(BootstrapMixin, forms.ModelForm):
  2773. tags = TagField(
  2774. required=False
  2775. )
  2776. class Meta:
  2777. model = FrontPort
  2778. fields = [
  2779. 'device', 'name', 'type', 'rear_port', 'rear_port_position', 'description', 'tags',
  2780. ]
  2781. widgets = {
  2782. 'device': forms.HiddenInput(),
  2783. 'type': StaticSelect2(),
  2784. 'rear_port': StaticSelect2(),
  2785. }
  2786. def __init__(self, *args, **kwargs):
  2787. super().__init__(*args, **kwargs)
  2788. # Limit RearPort choices to the local device
  2789. if hasattr(self.instance, 'device'):
  2790. self.fields['rear_port'].queryset = self.fields['rear_port'].queryset.filter(
  2791. device=self.instance.device
  2792. )
  2793. # TODO: Merge with FrontPortTemplateCreateForm to remove duplicate logic
  2794. class FrontPortCreateForm(BootstrapMixin, forms.Form):
  2795. device = DynamicModelChoiceField(
  2796. queryset=Device.objects.prefetch_related('device_type__manufacturer'),
  2797. widget=APISelect(
  2798. api_url="/api/dcim/devices/",
  2799. )
  2800. )
  2801. name_pattern = ExpandableNameField(
  2802. label='Name'
  2803. )
  2804. type = forms.ChoiceField(
  2805. choices=PortTypeChoices,
  2806. widget=StaticSelect2(),
  2807. )
  2808. rear_port_set = forms.MultipleChoiceField(
  2809. choices=[],
  2810. label='Rear ports',
  2811. help_text='Select one rear port assignment for each front port being created.',
  2812. )
  2813. description = forms.CharField(
  2814. required=False
  2815. )
  2816. def __init__(self, *args, **kwargs):
  2817. super().__init__(*args, **kwargs)
  2818. device = Device.objects.get(
  2819. pk=self.initial.get('device') or self.data.get('device')
  2820. )
  2821. # Determine which rear port positions are occupied. These will be excluded from the list of available
  2822. # mappings.
  2823. occupied_port_positions = [
  2824. (front_port.rear_port_id, front_port.rear_port_position)
  2825. for front_port in device.frontports.all()
  2826. ]
  2827. # Populate rear port choices
  2828. choices = []
  2829. rear_ports = RearPort.objects.filter(device=device)
  2830. for rear_port in rear_ports:
  2831. for i in range(1, rear_port.positions + 1):
  2832. if (rear_port.pk, i) not in occupied_port_positions:
  2833. choices.append(
  2834. ('{}:{}'.format(rear_port.pk, i), '{}:{}'.format(rear_port.name, i))
  2835. )
  2836. self.fields['rear_port_set'].choices = choices
  2837. def clean(self):
  2838. # Validate that the number of ports being created equals the number of selected (rear port, position) tuples
  2839. front_port_count = len(self.cleaned_data['name_pattern'])
  2840. rear_port_count = len(self.cleaned_data['rear_port_set'])
  2841. if front_port_count != rear_port_count:
  2842. raise forms.ValidationError({
  2843. 'rear_port_set': 'The provided name pattern will create {} ports, however {} rear port assignments '
  2844. 'were selected. These counts must match.'.format(front_port_count, rear_port_count)
  2845. })
  2846. def get_iterative_data(self, iteration):
  2847. # Assign rear port and position from selected set
  2848. rear_port, position = self.cleaned_data['rear_port_set'][iteration].split(':')
  2849. return {
  2850. 'rear_port': int(rear_port),
  2851. 'rear_port_position': int(position),
  2852. }
  2853. class FrontPortCSVForm(forms.ModelForm):
  2854. device = FlexibleModelChoiceField(
  2855. queryset=Device.objects.all(),
  2856. to_field_name='name',
  2857. help_text='Name or ID of device',
  2858. error_messages={
  2859. 'invalid_choice': 'Device not found.',
  2860. }
  2861. )
  2862. rear_port = FlexibleModelChoiceField(
  2863. queryset=RearPort.objects.all(),
  2864. to_field_name='name',
  2865. help_text='Name or ID of Rear Port',
  2866. error_messages={
  2867. 'invalid_choice': 'Rear Port not found.',
  2868. }
  2869. )
  2870. type = CSVChoiceField(
  2871. choices=PortTypeChoices,
  2872. )
  2873. class Meta:
  2874. model = FrontPort
  2875. fields = FrontPort.csv_headers
  2876. def __init__(self, *args, **kwargs):
  2877. super().__init__(*args, **kwargs)
  2878. # Limit RearPort choices to those belonging to this device (or VC master)
  2879. if self.is_bound:
  2880. try:
  2881. device = self.fields['device'].to_python(self.data['device'])
  2882. except forms.ValidationError:
  2883. device = None
  2884. else:
  2885. try:
  2886. device = self.instance.device
  2887. except Device.DoesNotExist:
  2888. device = None
  2889. if device:
  2890. self.fields['rear_port'].queryset = RearPort.objects.filter(
  2891. device__in=[device, device.get_vc_master()]
  2892. )
  2893. else:
  2894. self.fields['rear_port'].queryset = RearPort.objects.none()
  2895. class FrontPortBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
  2896. pk = forms.ModelMultipleChoiceField(
  2897. queryset=FrontPort.objects.all(),
  2898. widget=forms.MultipleHiddenInput()
  2899. )
  2900. type = forms.ChoiceField(
  2901. choices=add_blank_choice(PortTypeChoices),
  2902. required=False,
  2903. widget=StaticSelect2()
  2904. )
  2905. description = forms.CharField(
  2906. max_length=100,
  2907. required=False
  2908. )
  2909. class Meta:
  2910. nullable_fields = [
  2911. 'description',
  2912. ]
  2913. class FrontPortBulkRenameForm(BulkRenameForm):
  2914. pk = forms.ModelMultipleChoiceField(
  2915. queryset=FrontPort.objects.all(),
  2916. widget=forms.MultipleHiddenInput
  2917. )
  2918. class FrontPortBulkDisconnectForm(ConfirmationForm):
  2919. pk = forms.ModelMultipleChoiceField(
  2920. queryset=FrontPort.objects.all(),
  2921. widget=forms.MultipleHiddenInput
  2922. )
  2923. #
  2924. # Rear pass-through ports
  2925. #
  2926. class RearPortFilterForm(DeviceComponentFilterForm):
  2927. model = RearPort
  2928. tag = TagFilterField(model)
  2929. class RearPortForm(BootstrapMixin, forms.ModelForm):
  2930. tags = TagField(
  2931. required=False
  2932. )
  2933. class Meta:
  2934. model = RearPort
  2935. fields = [
  2936. 'device', 'name', 'type', 'positions', 'description', 'tags',
  2937. ]
  2938. widgets = {
  2939. 'device': forms.HiddenInput(),
  2940. 'type': StaticSelect2(),
  2941. }
  2942. class RearPortCreateForm(BootstrapMixin, forms.Form):
  2943. device = DynamicModelChoiceField(
  2944. queryset=Device.objects.prefetch_related('device_type__manufacturer'),
  2945. widget=APISelect(
  2946. api_url="/api/dcim/devices/",
  2947. )
  2948. )
  2949. name_pattern = ExpandableNameField(
  2950. label='Name'
  2951. )
  2952. type = forms.ChoiceField(
  2953. choices=PortTypeChoices,
  2954. widget=StaticSelect2(),
  2955. )
  2956. positions = forms.IntegerField(
  2957. min_value=REARPORT_POSITIONS_MIN,
  2958. max_value=REARPORT_POSITIONS_MAX,
  2959. initial=1,
  2960. help_text='The number of front ports which may be mapped to each rear port'
  2961. )
  2962. description = forms.CharField(
  2963. required=False
  2964. )
  2965. class RearPortCSVForm(forms.ModelForm):
  2966. device = FlexibleModelChoiceField(
  2967. queryset=Device.objects.all(),
  2968. to_field_name='name',
  2969. help_text='Name or ID of device',
  2970. error_messages={
  2971. 'invalid_choice': 'Device not found.',
  2972. }
  2973. )
  2974. type = CSVChoiceField(
  2975. choices=PortTypeChoices,
  2976. )
  2977. class Meta:
  2978. model = RearPort
  2979. fields = RearPort.csv_headers
  2980. class RearPortBulkEditForm(BootstrapMixin, AddRemoveTagsForm, BulkEditForm):
  2981. pk = forms.ModelMultipleChoiceField(
  2982. queryset=RearPort.objects.all(),
  2983. widget=forms.MultipleHiddenInput()
  2984. )
  2985. type = forms.ChoiceField(
  2986. choices=add_blank_choice(PortTypeChoices),
  2987. required=False,
  2988. widget=StaticSelect2()
  2989. )
  2990. description = forms.CharField(
  2991. max_length=100,
  2992. required=False
  2993. )
  2994. class Meta:
  2995. nullable_fields = [
  2996. 'description',
  2997. ]
  2998. class RearPortBulkRenameForm(BulkRenameForm):
  2999. pk = forms.ModelMultipleChoiceField(
  3000. queryset=RearPort.objects.all(),
  3001. widget=forms.MultipleHiddenInput
  3002. )
  3003. class RearPortBulkDisconnectForm(ConfirmationForm):
  3004. pk = forms.ModelMultipleChoiceField(
  3005. queryset=RearPort.objects.all(),
  3006. widget=forms.MultipleHiddenInput
  3007. )
  3008. #
  3009. # Cables
  3010. #
  3011. class ConnectCableToDeviceForm(BootstrapMixin, forms.ModelForm):
  3012. """
  3013. Base form for connecting a Cable to a Device component
  3014. """
  3015. termination_b_site = DynamicModelChoiceField(
  3016. queryset=Site.objects.all(),
  3017. label='Site',
  3018. required=False,
  3019. widget=APISelect(
  3020. api_url='/api/dcim/sites/',
  3021. filter_for={
  3022. 'termination_b_rack': 'site_id',
  3023. 'termination_b_device': 'site_id',
  3024. }
  3025. )
  3026. )
  3027. termination_b_rack = DynamicModelChoiceField(
  3028. queryset=Rack.objects.all(),
  3029. label='Rack',
  3030. required=False,
  3031. widget=APISelect(
  3032. api_url='/api/dcim/racks/',
  3033. filter_for={
  3034. 'termination_b_device': 'rack_id',
  3035. },
  3036. attrs={
  3037. 'nullable': 'true',
  3038. }
  3039. )
  3040. )
  3041. termination_b_device = DynamicModelChoiceField(
  3042. queryset=Device.objects.all(),
  3043. label='Device',
  3044. required=False,
  3045. widget=APISelect(
  3046. api_url='/api/dcim/devices/',
  3047. display_field='display_name',
  3048. filter_for={
  3049. 'termination_b_id': 'device_id',
  3050. }
  3051. )
  3052. )
  3053. class Meta:
  3054. model = Cable
  3055. fields = [
  3056. 'termination_b_site', 'termination_b_rack', 'termination_b_device', 'termination_b_id', 'type', 'status',
  3057. 'label', 'color', 'length', 'length_unit',
  3058. ]
  3059. widgets = {
  3060. 'status': StaticSelect2,
  3061. 'type': StaticSelect2,
  3062. 'length_unit': StaticSelect2,
  3063. }
  3064. class ConnectCableToConsolePortForm(ConnectCableToDeviceForm):
  3065. termination_b_id = forms.IntegerField(
  3066. label='Name',
  3067. widget=APISelect(
  3068. api_url='/api/dcim/console-ports/',
  3069. disabled_indicator='cable',
  3070. )
  3071. )
  3072. class ConnectCableToConsoleServerPortForm(ConnectCableToDeviceForm):
  3073. termination_b_id = forms.IntegerField(
  3074. label='Name',
  3075. widget=APISelect(
  3076. api_url='/api/dcim/console-server-ports/',
  3077. disabled_indicator='cable',
  3078. )
  3079. )
  3080. class ConnectCableToPowerPortForm(ConnectCableToDeviceForm):
  3081. termination_b_id = forms.IntegerField(
  3082. label='Name',
  3083. widget=APISelect(
  3084. api_url='/api/dcim/power-ports/',
  3085. disabled_indicator='cable',
  3086. )
  3087. )
  3088. class ConnectCableToPowerOutletForm(ConnectCableToDeviceForm):
  3089. termination_b_id = forms.IntegerField(
  3090. label='Name',
  3091. widget=APISelect(
  3092. api_url='/api/dcim/power-outlets/',
  3093. disabled_indicator='cable',
  3094. )
  3095. )
  3096. class ConnectCableToInterfaceForm(ConnectCableToDeviceForm):
  3097. termination_b_id = forms.IntegerField(
  3098. label='Name',
  3099. widget=APISelect(
  3100. api_url='/api/dcim/interfaces/',
  3101. disabled_indicator='cable',
  3102. additional_query_params={
  3103. 'kind': 'physical',
  3104. }
  3105. )
  3106. )
  3107. class ConnectCableToFrontPortForm(ConnectCableToDeviceForm):
  3108. termination_b_id = forms.IntegerField(
  3109. label='Name',
  3110. widget=APISelect(
  3111. api_url='/api/dcim/front-ports/',
  3112. disabled_indicator='cable',
  3113. )
  3114. )
  3115. class ConnectCableToRearPortForm(ConnectCableToDeviceForm):
  3116. termination_b_id = forms.IntegerField(
  3117. label='Name',
  3118. widget=APISelect(
  3119. api_url='/api/dcim/rear-ports/',
  3120. disabled_indicator='cable',
  3121. )
  3122. )
  3123. class ConnectCableToCircuitTerminationForm(BootstrapMixin, forms.ModelForm):
  3124. termination_b_provider = DynamicModelChoiceField(
  3125. queryset=Provider.objects.all(),
  3126. label='Provider',
  3127. required=False,
  3128. widget=APISelect(
  3129. api_url='/api/circuits/providers/',
  3130. filter_for={
  3131. 'termination_b_circuit': 'provider_id',
  3132. }
  3133. )
  3134. )
  3135. termination_b_site = DynamicModelChoiceField(
  3136. queryset=Site.objects.all(),
  3137. label='Site',
  3138. required=False,
  3139. widget=APISelect(
  3140. api_url='/api/dcim/sites/',
  3141. filter_for={
  3142. 'termination_b_circuit': 'site_id',
  3143. }
  3144. )
  3145. )
  3146. termination_b_circuit = DynamicModelChoiceField(
  3147. queryset=Circuit.objects.all(),
  3148. label='Circuit',
  3149. widget=APISelect(
  3150. api_url='/api/circuits/circuits/',
  3151. display_field='cid',
  3152. filter_for={
  3153. 'termination_b_id': 'circuit_id',
  3154. }
  3155. )
  3156. )
  3157. termination_b_id = forms.IntegerField(
  3158. label='Side',
  3159. widget=APISelect(
  3160. api_url='/api/circuits/circuit-terminations/',
  3161. disabled_indicator='cable',
  3162. display_field='term_side',
  3163. full=True
  3164. )
  3165. )
  3166. class Meta:
  3167. model = Cable
  3168. fields = [
  3169. 'termination_b_provider', 'termination_b_site', 'termination_b_circuit', 'termination_b_id', 'type',
  3170. 'status', 'label', 'color', 'length', 'length_unit',
  3171. ]
  3172. class ConnectCableToPowerFeedForm(BootstrapMixin, forms.ModelForm):
  3173. termination_b_site = DynamicModelChoiceField(
  3174. queryset=Site.objects.all(),
  3175. label='Site',
  3176. required=False,
  3177. widget=APISelect(
  3178. api_url='/api/dcim/sites/',
  3179. display_field='cid',
  3180. filter_for={
  3181. 'termination_b_rackgroup': 'site_id',
  3182. 'termination_b_powerpanel': 'site_id',
  3183. }
  3184. )
  3185. )
  3186. termination_b_rackgroup = DynamicModelChoiceField(
  3187. queryset=RackGroup.objects.all(),
  3188. label='Rack Group',
  3189. required=False,
  3190. widget=APISelect(
  3191. api_url='/api/dcim/rack-groups/',
  3192. display_field='cid',
  3193. filter_for={
  3194. 'termination_b_powerpanel': 'rackgroup_id',
  3195. }
  3196. )
  3197. )
  3198. termination_b_powerpanel = DynamicModelChoiceField(
  3199. queryset=PowerPanel.objects.all(),
  3200. label='Power Panel',
  3201. required=False,
  3202. widget=APISelect(
  3203. api_url='/api/dcim/power-panels/',
  3204. filter_for={
  3205. 'termination_b_id': 'power_panel_id',
  3206. }
  3207. )
  3208. )
  3209. termination_b_id = forms.IntegerField(
  3210. label='Name',
  3211. widget=APISelect(
  3212. api_url='/api/dcim/power-feeds/',
  3213. )
  3214. )
  3215. class Meta:
  3216. model = Cable
  3217. fields = [
  3218. 'termination_b_rackgroup', 'termination_b_powerpanel', 'termination_b_id', 'type', 'status', 'label',
  3219. 'color', 'length', 'length_unit',
  3220. ]
  3221. class CableForm(BootstrapMixin, forms.ModelForm):
  3222. class Meta:
  3223. model = Cable
  3224. fields = [
  3225. 'type', 'status', 'label', 'color', 'length', 'length_unit',
  3226. ]
  3227. widgets = {
  3228. 'status': StaticSelect2,
  3229. 'type': StaticSelect2,
  3230. 'length_unit': StaticSelect2,
  3231. }
  3232. class CableCSVForm(forms.ModelForm):
  3233. # Termination A
  3234. side_a_device = FlexibleModelChoiceField(
  3235. queryset=Device.objects.all(),
  3236. to_field_name='name',
  3237. help_text='Side A device name or ID',
  3238. error_messages={
  3239. 'invalid_choice': 'Side A device not found',
  3240. }
  3241. )
  3242. side_a_type = forms.ModelChoiceField(
  3243. queryset=ContentType.objects.all(),
  3244. limit_choices_to=CABLE_TERMINATION_MODELS,
  3245. to_field_name='model',
  3246. help_text='Side A type'
  3247. )
  3248. side_a_name = forms.CharField(
  3249. help_text='Side A component'
  3250. )
  3251. # Termination B
  3252. side_b_device = FlexibleModelChoiceField(
  3253. queryset=Device.objects.all(),
  3254. to_field_name='name',
  3255. help_text='Side B device name or ID',
  3256. error_messages={
  3257. 'invalid_choice': 'Side B device not found',
  3258. }
  3259. )
  3260. side_b_type = forms.ModelChoiceField(
  3261. queryset=ContentType.objects.all(),
  3262. limit_choices_to=CABLE_TERMINATION_MODELS,
  3263. to_field_name='model',
  3264. help_text='Side B type'
  3265. )
  3266. side_b_name = forms.CharField(
  3267. help_text='Side B component'
  3268. )
  3269. # Cable attributes
  3270. status = CSVChoiceField(
  3271. choices=CableStatusChoices,
  3272. required=False,
  3273. help_text='Connection status'
  3274. )
  3275. type = CSVChoiceField(
  3276. choices=CableTypeChoices,
  3277. required=False,
  3278. help_text='Cable type'
  3279. )
  3280. length_unit = CSVChoiceField(
  3281. choices=CableLengthUnitChoices,
  3282. required=False,
  3283. help_text='Length unit'
  3284. )
  3285. class Meta:
  3286. model = Cable
  3287. fields = [
  3288. 'side_a_device', 'side_a_type', 'side_a_name', 'side_b_device', 'side_b_type', 'side_b_name', 'type',
  3289. 'status', 'label', 'color', 'length', 'length_unit',
  3290. ]
  3291. help_texts = {
  3292. 'color': 'RGB color in hexadecimal (e.g. 00ff00)'
  3293. }
  3294. # TODO: Merge the clean() methods for either end
  3295. def clean_side_a_name(self):
  3296. device = self.cleaned_data.get('side_a_device')
  3297. content_type = self.cleaned_data.get('side_a_type')
  3298. name = self.cleaned_data.get('side_a_name')
  3299. if not device or not content_type or not name:
  3300. return None
  3301. model = content_type.model_class()
  3302. try:
  3303. termination_object = model.objects.get(
  3304. device=device,
  3305. name=name
  3306. )
  3307. if termination_object.cable is not None:
  3308. raise forms.ValidationError(
  3309. "Side A: {} {} is already connected".format(device, termination_object)
  3310. )
  3311. except ObjectDoesNotExist:
  3312. raise forms.ValidationError(
  3313. "A side termination not found: {} {}".format(device, name)
  3314. )
  3315. self.instance.termination_a = termination_object
  3316. return termination_object
  3317. def clean_side_b_name(self):
  3318. device = self.cleaned_data.get('side_b_device')
  3319. content_type = self.cleaned_data.get('side_b_type')
  3320. name = self.cleaned_data.get('side_b_name')
  3321. if not device or not content_type or not name:
  3322. return None
  3323. model = content_type.model_class()
  3324. try:
  3325. termination_object = model.objects.get(
  3326. device=device,
  3327. name=name
  3328. )
  3329. if termination_object.cable is not None:
  3330. raise forms.ValidationError(
  3331. "Side B: {} {} is already connected".format(device, termination_object)
  3332. )
  3333. except ObjectDoesNotExist:
  3334. raise forms.ValidationError(
  3335. "B side termination not found: {} {}".format(device, name)
  3336. )
  3337. self.instance.termination_b = termination_object
  3338. return termination_object
  3339. def clean_length_unit(self):
  3340. # Avoid trying to save as NULL
  3341. length_unit = self.cleaned_data.get('length_unit', None)
  3342. return length_unit if length_unit is not None else ''
  3343. class CableBulkEditForm(BootstrapMixin, BulkEditForm):
  3344. pk = forms.ModelMultipleChoiceField(
  3345. queryset=Cable.objects.all(),
  3346. widget=forms.MultipleHiddenInput
  3347. )
  3348. type = forms.ChoiceField(
  3349. choices=add_blank_choice(CableTypeChoices),
  3350. required=False,
  3351. initial='',
  3352. widget=StaticSelect2()
  3353. )
  3354. status = forms.ChoiceField(
  3355. choices=add_blank_choice(CableStatusChoices),
  3356. required=False,
  3357. widget=StaticSelect2(),
  3358. initial=''
  3359. )
  3360. label = forms.CharField(
  3361. max_length=100,
  3362. required=False
  3363. )
  3364. color = forms.CharField(
  3365. max_length=6, # RGB color code
  3366. required=False,
  3367. widget=ColorSelect()
  3368. )
  3369. length = forms.IntegerField(
  3370. min_value=1,
  3371. required=False
  3372. )
  3373. length_unit = forms.ChoiceField(
  3374. choices=add_blank_choice(CableLengthUnitChoices),
  3375. required=False,
  3376. initial='',
  3377. widget=StaticSelect2()
  3378. )
  3379. class Meta:
  3380. nullable_fields = [
  3381. 'type', 'status', 'label', 'color', 'length',
  3382. ]
  3383. def clean(self):
  3384. # Validate length/unit
  3385. length = self.cleaned_data.get('length')
  3386. length_unit = self.cleaned_data.get('length_unit')
  3387. if length and not length_unit:
  3388. raise forms.ValidationError({
  3389. 'length_unit': "Must specify a unit when setting length"
  3390. })
  3391. class CableFilterForm(BootstrapMixin, forms.Form):
  3392. model = Cable
  3393. q = forms.CharField(
  3394. required=False,
  3395. label='Search'
  3396. )
  3397. site = DynamicModelMultipleChoiceField(
  3398. queryset=Site.objects.all(),
  3399. to_field_name='slug',
  3400. required=False,
  3401. widget=APISelectMultiple(
  3402. api_url="/api/dcim/sites/",
  3403. value_field="slug",
  3404. filter_for={
  3405. 'rack_id': 'site',
  3406. 'device_id': 'site',
  3407. }
  3408. )
  3409. )
  3410. tenant = DynamicModelMultipleChoiceField(
  3411. queryset=Tenant.objects.all(),
  3412. to_field_name='slug',
  3413. required=False,
  3414. widget=APISelectMultiple(
  3415. api_url="/api/tenancy/tenants/",
  3416. value_field='slug',
  3417. filter_for={
  3418. 'device_id': 'tenant',
  3419. }
  3420. )
  3421. )
  3422. rack_id = DynamicModelMultipleChoiceField(
  3423. queryset=Rack.objects.all(),
  3424. required=False,
  3425. label='Rack',
  3426. widget=APISelectMultiple(
  3427. api_url="/api/dcim/racks/",
  3428. null_option=True,
  3429. filter_for={
  3430. 'device_id': 'rack_id',
  3431. }
  3432. )
  3433. )
  3434. type = forms.MultipleChoiceField(
  3435. choices=add_blank_choice(CableTypeChoices),
  3436. required=False,
  3437. widget=StaticSelect2()
  3438. )
  3439. status = forms.ChoiceField(
  3440. required=False,
  3441. choices=add_blank_choice(CableStatusChoices),
  3442. widget=StaticSelect2()
  3443. )
  3444. color = forms.CharField(
  3445. max_length=6, # RGB color code
  3446. required=False,
  3447. widget=ColorSelect()
  3448. )
  3449. device_id = DynamicModelMultipleChoiceField(
  3450. queryset=Device.objects.all(),
  3451. required=False,
  3452. label='Device',
  3453. widget=APISelectMultiple(
  3454. api_url='/api/dcim/devices/',
  3455. )
  3456. )
  3457. #
  3458. # Device bays
  3459. #
  3460. class DeviceBayFilterForm(DeviceComponentFilterForm):
  3461. model = DeviceBay
  3462. tag = TagFilterField(model)
  3463. class DeviceBayForm(BootstrapMixin, forms.ModelForm):
  3464. tags = TagField(
  3465. required=False
  3466. )
  3467. class Meta:
  3468. model = DeviceBay
  3469. fields = [
  3470. 'device', 'name', 'description', 'tags',
  3471. ]
  3472. widgets = {
  3473. 'device': forms.HiddenInput(),
  3474. }
  3475. class DeviceBayCreateForm(BootstrapMixin, forms.Form):
  3476. device = DynamicModelChoiceField(
  3477. queryset=Device.objects.prefetch_related('device_type__manufacturer'),
  3478. widget=APISelect(
  3479. api_url="/api/dcim/devices/",
  3480. )
  3481. )
  3482. name_pattern = ExpandableNameField(
  3483. label='Name'
  3484. )
  3485. tags = TagField(
  3486. required=False
  3487. )
  3488. class PopulateDeviceBayForm(BootstrapMixin, forms.Form):
  3489. installed_device = forms.ModelChoiceField(
  3490. queryset=Device.objects.all(),
  3491. label='Child Device',
  3492. help_text="Child devices must first be created and assigned to the site/rack of the parent device.",
  3493. widget=StaticSelect2(),
  3494. )
  3495. def __init__(self, device_bay, *args, **kwargs):
  3496. super().__init__(*args, **kwargs)
  3497. self.fields['installed_device'].queryset = Device.objects.filter(
  3498. site=device_bay.device.site,
  3499. rack=device_bay.device.rack,
  3500. parent_bay__isnull=True,
  3501. device_type__u_height=0,
  3502. device_type__subdevice_role=SubdeviceRoleChoices.ROLE_CHILD
  3503. ).exclude(pk=device_bay.device.pk)
  3504. class DeviceBayCSVForm(forms.ModelForm):
  3505. device = FlexibleModelChoiceField(
  3506. queryset=Device.objects.all(),
  3507. to_field_name='name',
  3508. help_text='Name or ID of device',
  3509. error_messages={
  3510. 'invalid_choice': 'Device not found.',
  3511. }
  3512. )
  3513. installed_device = FlexibleModelChoiceField(
  3514. queryset=Device.objects.all(),
  3515. required=False,
  3516. to_field_name='name',
  3517. help_text='Name or ID of device',
  3518. error_messages={
  3519. 'invalid_choice': 'Child device not found.',
  3520. }
  3521. )
  3522. class Meta:
  3523. model = DeviceBay
  3524. fields = DeviceBay.csv_headers
  3525. def __init__(self, *args, **kwargs):
  3526. super().__init__(*args, **kwargs)
  3527. # Limit installed device choices to devices of the correct type and location
  3528. if self.is_bound:
  3529. try:
  3530. device = self.fields['device'].to_python(self.data['device'])
  3531. except forms.ValidationError:
  3532. device = None
  3533. else:
  3534. try:
  3535. device = self.instance.device
  3536. except Device.DoesNotExist:
  3537. device = None
  3538. if device:
  3539. self.fields['installed_device'].queryset = Device.objects.filter(
  3540. site=device.site,
  3541. rack=device.rack,
  3542. parent_bay__isnull=True,
  3543. device_type__u_height=0,
  3544. device_type__subdevice_role=SubdeviceRoleChoices.ROLE_CHILD
  3545. ).exclude(pk=device.pk)
  3546. else:
  3547. self.fields['installed_device'].queryset = Interface.objects.none()
  3548. class DeviceBayBulkRenameForm(BulkRenameForm):
  3549. pk = forms.ModelMultipleChoiceField(
  3550. queryset=DeviceBay.objects.all(),
  3551. widget=forms.MultipleHiddenInput()
  3552. )
  3553. #
  3554. # Connections
  3555. #
  3556. class ConsoleConnectionFilterForm(BootstrapMixin, forms.Form):
  3557. site = DynamicModelMultipleChoiceField(
  3558. queryset=Site.objects.all(),
  3559. to_field_name='slug',
  3560. required=False,
  3561. widget=APISelectMultiple(
  3562. api_url="/api/dcim/sites/",
  3563. value_field="slug",
  3564. filter_for={
  3565. 'device_id': 'site',
  3566. }
  3567. )
  3568. )
  3569. device_id = DynamicModelMultipleChoiceField(
  3570. queryset=Device.objects.all(),
  3571. required=False,
  3572. label='Device',
  3573. widget=APISelectMultiple(
  3574. api_url='/api/dcim/devices/',
  3575. )
  3576. )
  3577. class PowerConnectionFilterForm(BootstrapMixin, forms.Form):
  3578. site = DynamicModelMultipleChoiceField(
  3579. queryset=Site.objects.all(),
  3580. to_field_name='slug',
  3581. required=False,
  3582. widget=APISelectMultiple(
  3583. api_url="/api/dcim/sites/",
  3584. value_field="slug",
  3585. filter_for={
  3586. 'device_id': 'site',
  3587. }
  3588. )
  3589. )
  3590. device_id = DynamicModelMultipleChoiceField(
  3591. queryset=Device.objects.all(),
  3592. required=False,
  3593. label='Device',
  3594. widget=APISelectMultiple(
  3595. api_url='/api/dcim/devices/',
  3596. )
  3597. )
  3598. class InterfaceConnectionFilterForm(BootstrapMixin, forms.Form):
  3599. site = DynamicModelMultipleChoiceField(
  3600. queryset=Site.objects.all(),
  3601. to_field_name='slug',
  3602. required=False,
  3603. widget=APISelectMultiple(
  3604. api_url="/api/dcim/sites/",
  3605. value_field="slug",
  3606. filter_for={
  3607. 'device_id': 'site',
  3608. }
  3609. )
  3610. )
  3611. device_id = DynamicModelMultipleChoiceField(
  3612. queryset=Device.objects.all(),
  3613. required=False,
  3614. label='Device',
  3615. widget=APISelectMultiple(
  3616. api_url='/api/dcim/devices/',
  3617. )
  3618. )
  3619. #
  3620. # Inventory items
  3621. #
  3622. class InventoryItemForm(BootstrapMixin, forms.ModelForm):
  3623. device = DynamicModelChoiceField(
  3624. queryset=Device.objects.prefetch_related('device_type__manufacturer'),
  3625. widget=APISelect(
  3626. api_url="/api/dcim/devices/"
  3627. )
  3628. )
  3629. manufacturer = DynamicModelChoiceField(
  3630. queryset=Manufacturer.objects.all(),
  3631. required=False,
  3632. widget=APISelect(
  3633. api_url="/api/dcim/manufacturers/"
  3634. )
  3635. )
  3636. tags = TagField(
  3637. required=False
  3638. )
  3639. class Meta:
  3640. model = InventoryItem
  3641. fields = [
  3642. 'name', 'device', 'manufacturer', 'part_id', 'serial', 'asset_tag', 'description', 'tags',
  3643. ]
  3644. class InventoryItemCreateForm(BootstrapMixin, forms.Form):
  3645. device = DynamicModelChoiceField(
  3646. queryset=Device.objects.prefetch_related('device_type__manufacturer'),
  3647. widget=APISelect(
  3648. api_url="/api/dcim/devices/",
  3649. )
  3650. )
  3651. name_pattern = ExpandableNameField(
  3652. label='Name'
  3653. )
  3654. manufacturer = DynamicModelChoiceField(
  3655. queryset=Manufacturer.objects.all(),
  3656. required=False,
  3657. widget=APISelect(
  3658. api_url="/api/dcim/manufacturers/"
  3659. )
  3660. )
  3661. part_id = forms.CharField(
  3662. max_length=50,
  3663. required=False,
  3664. label='Part ID'
  3665. )
  3666. serial = forms.CharField(
  3667. max_length=50,
  3668. required=False,
  3669. )
  3670. asset_tag = forms.CharField(
  3671. max_length=50,
  3672. required=False,
  3673. )
  3674. description = forms.CharField(
  3675. max_length=100,
  3676. required=False
  3677. )
  3678. class InventoryItemCSVForm(forms.ModelForm):
  3679. device = FlexibleModelChoiceField(
  3680. queryset=Device.objects.all(),
  3681. to_field_name='name',
  3682. help_text='Device name or ID',
  3683. error_messages={
  3684. 'invalid_choice': 'Device not found.',
  3685. }
  3686. )
  3687. manufacturer = forms.ModelChoiceField(
  3688. queryset=Manufacturer.objects.all(),
  3689. to_field_name='name',
  3690. required=False,
  3691. help_text='Manufacturer name',
  3692. error_messages={
  3693. 'invalid_choice': 'Invalid manufacturer.',
  3694. }
  3695. )
  3696. class Meta:
  3697. model = InventoryItem
  3698. fields = InventoryItem.csv_headers
  3699. class InventoryItemBulkEditForm(BootstrapMixin, BulkEditForm):
  3700. pk = forms.ModelMultipleChoiceField(
  3701. queryset=InventoryItem.objects.all(),
  3702. widget=forms.MultipleHiddenInput()
  3703. )
  3704. device = DynamicModelChoiceField(
  3705. queryset=Device.objects.all(),
  3706. required=False,
  3707. widget=APISelect(
  3708. api_url="/api/dcim/devices/"
  3709. )
  3710. )
  3711. manufacturer = DynamicModelChoiceField(
  3712. queryset=Manufacturer.objects.all(),
  3713. required=False,
  3714. widget=APISelect(
  3715. api_url="/api/dcim/manufacturers/"
  3716. )
  3717. )
  3718. part_id = forms.CharField(
  3719. max_length=50,
  3720. required=False,
  3721. label='Part ID'
  3722. )
  3723. description = forms.CharField(
  3724. max_length=100,
  3725. required=False
  3726. )
  3727. class Meta:
  3728. nullable_fields = [
  3729. 'manufacturer', 'part_id', 'description',
  3730. ]
  3731. class InventoryItemFilterForm(BootstrapMixin, forms.Form):
  3732. model = InventoryItem
  3733. q = forms.CharField(
  3734. required=False,
  3735. label='Search'
  3736. )
  3737. region = DynamicModelMultipleChoiceField(
  3738. queryset=Region.objects.all(),
  3739. to_field_name='slug',
  3740. required=False,
  3741. widget=APISelectMultiple(
  3742. api_url="/api/dcim/regions/",
  3743. value_field="slug",
  3744. filter_for={
  3745. 'site': 'region'
  3746. }
  3747. )
  3748. )
  3749. site = DynamicModelMultipleChoiceField(
  3750. queryset=Site.objects.all(),
  3751. to_field_name='slug',
  3752. required=False,
  3753. widget=APISelectMultiple(
  3754. api_url="/api/dcim/sites/",
  3755. value_field="slug",
  3756. filter_for={
  3757. 'device_id': 'site'
  3758. }
  3759. )
  3760. )
  3761. device_id = DynamicModelMultipleChoiceField(
  3762. queryset=Device.objects.all(),
  3763. required=False,
  3764. label='Device',
  3765. widget=APISelect(
  3766. api_url='/api/dcim/devices/',
  3767. )
  3768. )
  3769. manufacturer = DynamicModelMultipleChoiceField(
  3770. queryset=Manufacturer.objects.all(),
  3771. to_field_name='slug',
  3772. required=False,
  3773. widget=APISelect(
  3774. api_url="/api/dcim/manufacturers/",
  3775. value_field="slug",
  3776. )
  3777. )
  3778. discovered = forms.NullBooleanField(
  3779. required=False,
  3780. widget=StaticSelect2(
  3781. choices=BOOLEAN_WITH_BLANK_CHOICES
  3782. )
  3783. )
  3784. tag = TagFilterField(model)
  3785. #
  3786. # Virtual chassis
  3787. #
  3788. class DeviceSelectionForm(forms.Form):
  3789. pk = forms.ModelMultipleChoiceField(
  3790. queryset=Device.objects.all(),
  3791. widget=forms.MultipleHiddenInput()
  3792. )
  3793. class VirtualChassisForm(BootstrapMixin, forms.ModelForm):
  3794. tags = TagField(
  3795. required=False
  3796. )
  3797. class Meta:
  3798. model = VirtualChassis
  3799. fields = [
  3800. 'master', 'domain', 'tags',
  3801. ]
  3802. widgets = {
  3803. 'master': SelectWithPK(),
  3804. }
  3805. class BaseVCMemberFormSet(forms.BaseModelFormSet):
  3806. def clean(self):
  3807. super().clean()
  3808. # Check for duplicate VC position values
  3809. vc_position_list = []
  3810. for form in self.forms:
  3811. vc_position = form.cleaned_data.get('vc_position')
  3812. if vc_position:
  3813. if vc_position in vc_position_list:
  3814. error_msg = 'A virtual chassis member already exists in position {}.'.format(vc_position)
  3815. form.add_error('vc_position', error_msg)
  3816. vc_position_list.append(vc_position)
  3817. class DeviceVCMembershipForm(forms.ModelForm):
  3818. class Meta:
  3819. model = Device
  3820. fields = [
  3821. 'vc_position', 'vc_priority',
  3822. ]
  3823. labels = {
  3824. 'vc_position': 'Position',
  3825. 'vc_priority': 'Priority',
  3826. }
  3827. def __init__(self, validate_vc_position=False, *args, **kwargs):
  3828. super().__init__(*args, **kwargs)
  3829. # Require VC position (only required when the Device is a VirtualChassis member)
  3830. self.fields['vc_position'].required = True
  3831. # Validation of vc_position is optional. This is only required when adding a new member to an existing
  3832. # VirtualChassis. Otherwise, vc_position validation is handled by BaseVCMemberFormSet.
  3833. self.validate_vc_position = validate_vc_position
  3834. def clean_vc_position(self):
  3835. vc_position = self.cleaned_data['vc_position']
  3836. if self.validate_vc_position:
  3837. conflicting_members = Device.objects.filter(
  3838. virtual_chassis=self.instance.virtual_chassis,
  3839. vc_position=vc_position
  3840. )
  3841. if conflicting_members.exists():
  3842. raise forms.ValidationError(
  3843. 'A virtual chassis member already exists in position {}.'.format(vc_position)
  3844. )
  3845. return vc_position
  3846. class VCMemberSelectForm(BootstrapMixin, forms.Form):
  3847. site = DynamicModelChoiceField(
  3848. queryset=Site.objects.all(),
  3849. required=False,
  3850. widget=APISelect(
  3851. api_url="/api/dcim/sites/",
  3852. filter_for={
  3853. 'rack': 'site_id',
  3854. 'device': 'site_id',
  3855. }
  3856. )
  3857. )
  3858. rack = DynamicModelChoiceField(
  3859. queryset=Rack.objects.all(),
  3860. required=False,
  3861. widget=APISelect(
  3862. api_url='/api/dcim/racks/',
  3863. filter_for={
  3864. 'device': 'rack_id'
  3865. },
  3866. attrs={
  3867. 'nullable': 'true',
  3868. }
  3869. )
  3870. )
  3871. device = DynamicModelChoiceField(
  3872. queryset=Device.objects.filter(
  3873. virtual_chassis__isnull=True
  3874. ),
  3875. widget=APISelect(
  3876. api_url='/api/dcim/devices/',
  3877. display_field='display_name',
  3878. disabled_indicator='virtual_chassis'
  3879. )
  3880. )
  3881. def clean_device(self):
  3882. device = self.cleaned_data['device']
  3883. if device.virtual_chassis is not None:
  3884. raise forms.ValidationError(
  3885. "Device {} is already assigned to a virtual chassis.".format(device)
  3886. )
  3887. return device
  3888. class VirtualChassisFilterForm(BootstrapMixin, CustomFieldFilterForm):
  3889. model = VirtualChassis
  3890. q = forms.CharField(
  3891. required=False,
  3892. label='Search'
  3893. )
  3894. region = DynamicModelMultipleChoiceField(
  3895. queryset=Region.objects.all(),
  3896. to_field_name='slug',
  3897. required=False,
  3898. widget=APISelectMultiple(
  3899. api_url="/api/dcim/regions/",
  3900. value_field="slug",
  3901. filter_for={
  3902. 'site': 'region'
  3903. }
  3904. )
  3905. )
  3906. site = DynamicModelMultipleChoiceField(
  3907. queryset=Site.objects.all(),
  3908. to_field_name='slug',
  3909. required=False,
  3910. widget=APISelectMultiple(
  3911. api_url="/api/dcim/sites/",
  3912. value_field="slug",
  3913. )
  3914. )
  3915. tenant_group = DynamicModelMultipleChoiceField(
  3916. queryset=TenantGroup.objects.all(),
  3917. to_field_name='slug',
  3918. required=False,
  3919. widget=APISelectMultiple(
  3920. api_url="/api/tenancy/tenant-groups/",
  3921. value_field="slug",
  3922. null_option=True,
  3923. filter_for={
  3924. 'tenant': 'group'
  3925. }
  3926. )
  3927. )
  3928. tenant = DynamicModelMultipleChoiceField(
  3929. queryset=Tenant.objects.all(),
  3930. to_field_name='slug',
  3931. required=False,
  3932. widget=APISelectMultiple(
  3933. api_url="/api/tenancy/tenants/",
  3934. value_field="slug",
  3935. null_option=True,
  3936. )
  3937. )
  3938. tag = TagFilterField(model)
  3939. #
  3940. # Power panels
  3941. #
  3942. class PowerPanelForm(BootstrapMixin, forms.ModelForm):
  3943. site = DynamicModelChoiceField(
  3944. queryset=Site.objects.all(),
  3945. required=False,
  3946. widget=APISelect(
  3947. api_url="/api/dcim/sites/",
  3948. filter_for={
  3949. 'rack_group': 'site_id',
  3950. }
  3951. )
  3952. )
  3953. rack_group = DynamicModelChoiceField(
  3954. queryset=RackGroup.objects.all(),
  3955. required=False,
  3956. widget=APISelect(
  3957. api_url='/api/dcim/rack-groups/',
  3958. )
  3959. )
  3960. class Meta:
  3961. model = PowerPanel
  3962. fields = [
  3963. 'site', 'rack_group', 'name',
  3964. ]
  3965. class PowerPanelCSVForm(forms.ModelForm):
  3966. site = forms.ModelChoiceField(
  3967. queryset=Site.objects.all(),
  3968. to_field_name='name',
  3969. help_text='Name of parent site',
  3970. error_messages={
  3971. 'invalid_choice': 'Site not found.',
  3972. }
  3973. )
  3974. rack_group_name = forms.CharField(
  3975. required=False,
  3976. help_text="Rack group name (optional)"
  3977. )
  3978. class Meta:
  3979. model = PowerPanel
  3980. fields = PowerPanel.csv_headers
  3981. def clean(self):
  3982. super().clean()
  3983. site = self.cleaned_data.get('site')
  3984. rack_group_name = self.cleaned_data.get('rack_group_name')
  3985. # Validate rack group
  3986. if rack_group_name:
  3987. try:
  3988. self.instance.rack_group = RackGroup.objects.get(site=site, name=rack_group_name)
  3989. except RackGroup.DoesNotExist:
  3990. raise forms.ValidationError(
  3991. "Rack group {} not found in site {}".format(rack_group_name, site)
  3992. )
  3993. class PowerPanelFilterForm(BootstrapMixin, CustomFieldFilterForm):
  3994. model = PowerPanel
  3995. q = forms.CharField(
  3996. required=False,
  3997. label='Search'
  3998. )
  3999. region = DynamicModelMultipleChoiceField(
  4000. queryset=Region.objects.all(),
  4001. to_field_name='slug',
  4002. required=False,
  4003. widget=APISelectMultiple(
  4004. api_url="/api/dcim/regions/",
  4005. value_field="slug",
  4006. filter_for={
  4007. 'site': 'region'
  4008. }
  4009. )
  4010. )
  4011. site = DynamicModelMultipleChoiceField(
  4012. queryset=Site.objects.all(),
  4013. to_field_name='slug',
  4014. required=False,
  4015. widget=APISelectMultiple(
  4016. api_url="/api/dcim/sites/",
  4017. value_field="slug",
  4018. filter_for={
  4019. 'rack_group_id': 'site',
  4020. }
  4021. )
  4022. )
  4023. rack_group_id = DynamicModelMultipleChoiceField(
  4024. queryset=RackGroup.objects.all(),
  4025. required=False,
  4026. label='Rack group (ID)',
  4027. widget=APISelectMultiple(
  4028. api_url="/api/dcim/rack-groups/",
  4029. null_option=True,
  4030. )
  4031. )
  4032. #
  4033. # Power feeds
  4034. #
  4035. class PowerFeedForm(BootstrapMixin, CustomFieldModelForm):
  4036. site = DynamicModelChoiceField(
  4037. queryset=Site.objects.all(),
  4038. required=False,
  4039. widget=APISelect(
  4040. api_url='/api/dcim/sites/',
  4041. filter_for={
  4042. 'power_panel': 'site_id',
  4043. 'rack': 'site_id',
  4044. }
  4045. )
  4046. )
  4047. power_panel = DynamicModelChoiceField(
  4048. queryset=PowerPanel.objects.all(),
  4049. widget=APISelect(
  4050. api_url="/api/dcim/power-panels/"
  4051. )
  4052. )
  4053. rack = DynamicModelChoiceField(
  4054. queryset=Rack.objects.all(),
  4055. required=False,
  4056. widget=APISelect(
  4057. api_url="/api/dcim/racks/"
  4058. )
  4059. )
  4060. comments = CommentField()
  4061. tags = TagField(
  4062. required=False
  4063. )
  4064. class Meta:
  4065. model = PowerFeed
  4066. fields = [
  4067. 'site', 'power_panel', 'rack', 'name', 'status', 'type', 'supply', 'phase', 'voltage', 'amperage',
  4068. 'max_utilization', 'comments', 'tags',
  4069. ]
  4070. widgets = {
  4071. 'status': StaticSelect2(),
  4072. 'type': StaticSelect2(),
  4073. 'supply': StaticSelect2(),
  4074. 'phase': StaticSelect2(),
  4075. }
  4076. def __init__(self, *args, **kwargs):
  4077. super().__init__(*args, **kwargs)
  4078. # Initialize site field
  4079. if self.instance and hasattr(self.instance, 'power_panel'):
  4080. self.initial['site'] = self.instance.power_panel.site
  4081. class PowerFeedCSVForm(CustomFieldModelCSVForm):
  4082. site = forms.ModelChoiceField(
  4083. queryset=Site.objects.all(),
  4084. to_field_name='name',
  4085. help_text='Name of parent site',
  4086. error_messages={
  4087. 'invalid_choice': 'Site not found.',
  4088. }
  4089. )
  4090. panel_name = forms.ModelChoiceField(
  4091. queryset=PowerPanel.objects.all(),
  4092. to_field_name='name',
  4093. help_text='Name of upstream power panel',
  4094. error_messages={
  4095. 'invalid_choice': 'Power panel not found.',
  4096. }
  4097. )
  4098. rack_group = forms.CharField(
  4099. required=False,
  4100. help_text="Rack group name (optional)"
  4101. )
  4102. rack_name = forms.CharField(
  4103. required=False,
  4104. help_text="Rack name (optional)"
  4105. )
  4106. status = CSVChoiceField(
  4107. choices=PowerFeedStatusChoices,
  4108. required=False,
  4109. help_text='Operational status'
  4110. )
  4111. type = CSVChoiceField(
  4112. choices=PowerFeedTypeChoices,
  4113. required=False,
  4114. help_text='Primary or redundant'
  4115. )
  4116. supply = CSVChoiceField(
  4117. choices=PowerFeedSupplyChoices,
  4118. required=False,
  4119. help_text='AC/DC'
  4120. )
  4121. phase = CSVChoiceField(
  4122. choices=PowerFeedPhaseChoices,
  4123. required=False,
  4124. help_text='Single or three-phase'
  4125. )
  4126. class Meta:
  4127. model = PowerFeed
  4128. fields = PowerFeed.csv_headers
  4129. def clean(self):
  4130. super().clean()
  4131. site = self.cleaned_data.get('site')
  4132. panel_name = self.cleaned_data.get('panel_name')
  4133. rack_group = self.cleaned_data.get('rack_group')
  4134. rack_name = self.cleaned_data.get('rack_name')
  4135. # Validate power panel
  4136. if panel_name:
  4137. try:
  4138. self.instance.power_panel = PowerPanel.objects.get(site=site, name=panel_name)
  4139. except Rack.DoesNotExist:
  4140. raise forms.ValidationError(
  4141. "Power panel {} not found in site {}".format(panel_name, site)
  4142. )
  4143. # Validate rack
  4144. if rack_name:
  4145. try:
  4146. self.instance.rack = Rack.objects.get(site=site, group__name=rack_group, name=rack_name)
  4147. except Rack.DoesNotExist:
  4148. raise forms.ValidationError(
  4149. "Rack {} not found in site {}, group {}".format(rack_name, site, rack_group)
  4150. )
  4151. class PowerFeedBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEditForm):
  4152. pk = forms.ModelMultipleChoiceField(
  4153. queryset=PowerFeed.objects.all(),
  4154. widget=forms.MultipleHiddenInput
  4155. )
  4156. power_panel = DynamicModelChoiceField(
  4157. queryset=PowerPanel.objects.all(),
  4158. required=False,
  4159. widget=APISelect(
  4160. api_url="/api/dcim/power-panels/",
  4161. filter_for={
  4162. 'rackgroup': 'site_id',
  4163. }
  4164. )
  4165. )
  4166. rack = DynamicModelChoiceField(
  4167. queryset=Rack.objects.all(),
  4168. required=False,
  4169. widget=APISelect(
  4170. api_url="/api/dcim/racks",
  4171. )
  4172. )
  4173. status = forms.ChoiceField(
  4174. choices=add_blank_choice(PowerFeedStatusChoices),
  4175. required=False,
  4176. initial='',
  4177. widget=StaticSelect2()
  4178. )
  4179. type = forms.ChoiceField(
  4180. choices=add_blank_choice(PowerFeedTypeChoices),
  4181. required=False,
  4182. initial='',
  4183. widget=StaticSelect2()
  4184. )
  4185. supply = forms.ChoiceField(
  4186. choices=add_blank_choice(PowerFeedSupplyChoices),
  4187. required=False,
  4188. initial='',
  4189. widget=StaticSelect2()
  4190. )
  4191. phase = forms.ChoiceField(
  4192. choices=add_blank_choice(PowerFeedPhaseChoices),
  4193. required=False,
  4194. initial='',
  4195. widget=StaticSelect2()
  4196. )
  4197. voltage = forms.IntegerField(
  4198. required=False
  4199. )
  4200. amperage = forms.IntegerField(
  4201. required=False
  4202. )
  4203. max_utilization = forms.IntegerField(
  4204. required=False
  4205. )
  4206. comments = CommentField(
  4207. widget=SmallTextarea,
  4208. label='Comments'
  4209. )
  4210. class Meta:
  4211. nullable_fields = [
  4212. 'rackgroup', 'comments',
  4213. ]
  4214. class PowerFeedFilterForm(BootstrapMixin, CustomFieldFilterForm):
  4215. model = PowerFeed
  4216. q = forms.CharField(
  4217. required=False,
  4218. label='Search'
  4219. )
  4220. region = DynamicModelMultipleChoiceField(
  4221. queryset=Region.objects.all(),
  4222. to_field_name='slug',
  4223. required=False,
  4224. widget=APISelectMultiple(
  4225. api_url="/api/dcim/regions/",
  4226. value_field="slug",
  4227. filter_for={
  4228. 'site': 'region'
  4229. }
  4230. )
  4231. )
  4232. site = DynamicModelMultipleChoiceField(
  4233. queryset=Site.objects.all(),
  4234. to_field_name='slug',
  4235. required=False,
  4236. widget=APISelectMultiple(
  4237. api_url="/api/dcim/sites/",
  4238. value_field="slug",
  4239. filter_for={
  4240. 'power_panel_id': 'site',
  4241. 'rack_id': 'site',
  4242. }
  4243. )
  4244. )
  4245. power_panel_id = DynamicModelMultipleChoiceField(
  4246. queryset=PowerPanel.objects.all(),
  4247. required=False,
  4248. label='Power panel',
  4249. widget=APISelectMultiple(
  4250. api_url="/api/dcim/power-panels/",
  4251. null_option=True,
  4252. )
  4253. )
  4254. rack_id = DynamicModelMultipleChoiceField(
  4255. queryset=Rack.objects.all(),
  4256. required=False,
  4257. label='Rack',
  4258. widget=APISelectMultiple(
  4259. api_url="/api/dcim/racks/",
  4260. null_option=True,
  4261. )
  4262. )
  4263. status = forms.MultipleChoiceField(
  4264. choices=PowerFeedStatusChoices,
  4265. required=False,
  4266. widget=StaticSelect2Multiple()
  4267. )
  4268. type = forms.ChoiceField(
  4269. choices=add_blank_choice(PowerFeedTypeChoices),
  4270. required=False,
  4271. widget=StaticSelect2()
  4272. )
  4273. supply = forms.ChoiceField(
  4274. choices=add_blank_choice(PowerFeedSupplyChoices),
  4275. required=False,
  4276. widget=StaticSelect2()
  4277. )
  4278. phase = forms.ChoiceField(
  4279. choices=add_blank_choice(PowerFeedPhaseChoices),
  4280. required=False,
  4281. widget=StaticSelect2()
  4282. )
  4283. voltage = forms.IntegerField(
  4284. required=False
  4285. )
  4286. amperage = forms.IntegerField(
  4287. required=False
  4288. )
  4289. max_utilization = forms.IntegerField(
  4290. required=False
  4291. )
  4292. tag = TagFilterField(model)