object_create.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359
  1. from django import forms
  2. from django.utils.translation import gettext_lazy as _
  3. from dcim.models import *
  4. from netbox.forms import NetBoxModelForm
  5. from netbox.forms.mixins import OwnerMixin
  6. from utilities.forms.fields import DynamicModelChoiceField, DynamicModelMultipleChoiceField, ExpandableNameField
  7. from utilities.forms.rendering import FieldSet, TabbedGroups
  8. from utilities.forms.widgets import APISelect
  9. from . import model_forms
  10. __all__ = (
  11. 'ComponentCreateForm',
  12. 'ConsolePortCreateForm',
  13. 'ConsolePortTemplateCreateForm',
  14. 'ConsoleServerPortCreateForm',
  15. 'ConsoleServerPortTemplateCreateForm',
  16. 'DeviceBayCreateForm',
  17. 'DeviceBayTemplateCreateForm',
  18. 'FrontPortCreateForm',
  19. 'FrontPortTemplateCreateForm',
  20. 'InterfaceCreateForm',
  21. 'InterfaceTemplateCreateForm',
  22. 'InventoryItemCreateForm',
  23. 'InventoryItemTemplateCreateForm',
  24. 'ModuleBayCreateForm',
  25. 'ModuleBayTemplateCreateForm',
  26. 'PowerOutletCreateForm',
  27. 'PowerOutletTemplateCreateForm',
  28. 'PowerPortCreateForm',
  29. 'PowerPortTemplateCreateForm',
  30. 'RearPortCreateForm',
  31. 'RearPortTemplateCreateForm',
  32. 'VirtualChassisCreateForm',
  33. )
  34. class ComponentCreateForm(forms.Form):
  35. """
  36. Subclass this form when facilitating the creation of one or more component or component template objects based on
  37. a name pattern.
  38. """
  39. name = ExpandableNameField(
  40. label=_('Name'),
  41. )
  42. label = ExpandableNameField(
  43. label=_('Label'),
  44. required=False,
  45. help_text=_('Alphanumeric ranges are supported. (Must match the number of objects being created.)')
  46. )
  47. # Identify the fields which support replication (i.e. ExpandableNameFields). This is referenced by
  48. # ComponentCreateView when creating objects.
  49. replication_fields = ('name', 'label')
  50. def clean(self):
  51. super().clean()
  52. # Validate that all replication fields generate an equal number of values (or a single value)
  53. if not (patterns := self.cleaned_data.get(self.replication_fields[0])):
  54. return
  55. pattern_count = len(patterns)
  56. for field_name in self.replication_fields:
  57. value_count = len(self.cleaned_data[field_name])
  58. if self.cleaned_data[field_name]:
  59. if value_count == 1:
  60. # If the field resolves to a single value (because no pattern was used), multiply it by the number
  61. # of expected values. This allows us to reuse the same label when creating multiple components.
  62. self.cleaned_data[field_name] = self.cleaned_data[field_name] * pattern_count
  63. elif value_count != pattern_count:
  64. raise forms.ValidationError({
  65. field_name: _(
  66. "The provided pattern specifies {value_count} values, but {pattern_count} are expected."
  67. ).format(value_count=value_count, pattern_count=pattern_count)
  68. }, code='label_pattern_mismatch')
  69. #
  70. # Device component templates
  71. #
  72. class ConsolePortTemplateCreateForm(ComponentCreateForm, model_forms.ConsolePortTemplateForm):
  73. class Meta(model_forms.ConsolePortTemplateForm.Meta):
  74. exclude = ('name', 'label')
  75. class ConsoleServerPortTemplateCreateForm(ComponentCreateForm, model_forms.ConsoleServerPortTemplateForm):
  76. class Meta(model_forms.ConsoleServerPortTemplateForm.Meta):
  77. exclude = ('name', 'label')
  78. class PowerPortTemplateCreateForm(ComponentCreateForm, model_forms.PowerPortTemplateForm):
  79. class Meta(model_forms.PowerPortTemplateForm.Meta):
  80. exclude = ('name', 'label')
  81. class PowerOutletTemplateCreateForm(ComponentCreateForm, model_forms.PowerOutletTemplateForm):
  82. class Meta(model_forms.PowerOutletTemplateForm.Meta):
  83. exclude = ('name', 'label')
  84. class InterfaceTemplateCreateForm(ComponentCreateForm, model_forms.InterfaceTemplateForm):
  85. class Meta(model_forms.InterfaceTemplateForm.Meta):
  86. exclude = ('name', 'label')
  87. class FrontPortTemplateCreateForm(ComponentCreateForm, model_forms.FrontPortTemplateForm):
  88. # Override fieldsets from FrontPortTemplateForm
  89. fieldsets = (
  90. FieldSet(
  91. TabbedGroups(
  92. FieldSet('device_type', name=_('Device Type')),
  93. FieldSet('module_type', name=_('Module Type')),
  94. ),
  95. 'name', 'label', 'type', 'color', 'positions', 'rear_ports', 'description',
  96. ),
  97. )
  98. class Meta:
  99. model = FrontPortTemplate
  100. fields = (
  101. 'device_type', 'module_type', 'type', 'color', 'positions', 'description',
  102. )
  103. def get_iterative_data(self, iteration):
  104. positions = self.cleaned_data['positions']
  105. offset = positions * iteration
  106. return {
  107. 'rear_ports': self.cleaned_data['rear_ports'][offset:offset + positions]
  108. }
  109. class RearPortTemplateCreateForm(ComponentCreateForm, model_forms.RearPortTemplateForm):
  110. class Meta(model_forms.RearPortTemplateForm.Meta):
  111. exclude = ('name', 'label')
  112. class DeviceBayTemplateCreateForm(ComponentCreateForm, model_forms.DeviceBayTemplateForm):
  113. class Meta(model_forms.DeviceBayTemplateForm.Meta):
  114. exclude = ('name', 'label')
  115. class ModuleBayTemplateCreateForm(ComponentCreateForm, model_forms.ModuleBayTemplateForm):
  116. position = ExpandableNameField(
  117. label=_('Position'),
  118. required=False,
  119. help_text=_('Alphanumeric ranges are supported. (Must match the number of objects being created.)')
  120. )
  121. replication_fields = ('name', 'label', 'position')
  122. class Meta(model_forms.ModuleBayTemplateForm.Meta):
  123. exclude = ('name', 'label', 'position')
  124. class InventoryItemTemplateCreateForm(ComponentCreateForm, model_forms.InventoryItemTemplateForm):
  125. class Meta(model_forms.InventoryItemTemplateForm.Meta):
  126. exclude = ('name', 'label')
  127. #
  128. # Device components
  129. #
  130. class ConsolePortCreateForm(ComponentCreateForm, model_forms.ConsolePortForm):
  131. class Meta(model_forms.ConsolePortForm.Meta):
  132. exclude = ('name', 'label')
  133. class ConsoleServerPortCreateForm(ComponentCreateForm, model_forms.ConsoleServerPortForm):
  134. class Meta(model_forms.ConsoleServerPortForm.Meta):
  135. exclude = ('name', 'label')
  136. class PowerPortCreateForm(ComponentCreateForm, model_forms.PowerPortForm):
  137. class Meta(model_forms.PowerPortForm.Meta):
  138. exclude = ('name', 'label')
  139. class PowerOutletCreateForm(ComponentCreateForm, model_forms.PowerOutletForm):
  140. class Meta(model_forms.PowerOutletForm.Meta):
  141. exclude = ('name', 'label')
  142. class InterfaceCreateForm(ComponentCreateForm, model_forms.InterfaceForm):
  143. class Meta(model_forms.InterfaceForm.Meta):
  144. exclude = ('name', 'label')
  145. class FrontPortCreateForm(ComponentCreateForm, model_forms.FrontPortForm):
  146. device = DynamicModelChoiceField(
  147. label=_('Device'),
  148. queryset=Device.objects.all(),
  149. selector=True,
  150. widget=APISelect(
  151. # TODO: Clean up the application of HTMXSelect attributes
  152. attrs={
  153. 'hx-get': '.',
  154. 'hx-include': '#form_fields',
  155. 'hx-target': '#form_fields',
  156. }
  157. )
  158. )
  159. # Override fieldsets from FrontPortForm to omit rear_port_position
  160. fieldsets = (
  161. FieldSet(
  162. 'device', 'module', 'name', 'label', 'type', 'color', 'positions', 'rear_ports', 'mark_connected',
  163. 'description', 'tags',
  164. ),
  165. )
  166. class Meta:
  167. model = FrontPort
  168. fields = [
  169. 'device', 'module', 'type', 'color', 'positions', 'mark_connected', 'description', 'owner', 'tags',
  170. ]
  171. def get_iterative_data(self, iteration):
  172. positions = self.cleaned_data['positions']
  173. offset = positions * iteration
  174. return {
  175. 'rear_ports': self.cleaned_data['rear_ports'][offset:offset + positions]
  176. }
  177. class RearPortCreateForm(ComponentCreateForm, model_forms.RearPortForm):
  178. class Meta(model_forms.RearPortForm.Meta):
  179. exclude = ('name', 'label')
  180. class DeviceBayCreateForm(ComponentCreateForm, model_forms.DeviceBayForm):
  181. class Meta(model_forms.DeviceBayForm.Meta):
  182. exclude = ('name', 'label')
  183. class ModuleBayCreateForm(ComponentCreateForm, model_forms.ModuleBayForm):
  184. position = ExpandableNameField(
  185. label=_('Position'),
  186. required=False,
  187. help_text=_('Alphanumeric ranges are supported. (Must match the number of objects being created.)')
  188. )
  189. replication_fields = ('name', 'label', 'position')
  190. class Meta(model_forms.ModuleBayForm.Meta):
  191. exclude = ('name', 'label', 'position')
  192. class InventoryItemCreateForm(ComponentCreateForm, model_forms.InventoryItemForm):
  193. class Meta(model_forms.InventoryItemForm.Meta):
  194. exclude = ('name', 'label')
  195. #
  196. # Virtual chassis
  197. #
  198. class VirtualChassisCreateForm(OwnerMixin, NetBoxModelForm):
  199. region = DynamicModelChoiceField(
  200. label=_('Region'),
  201. queryset=Region.objects.all(),
  202. required=False,
  203. initial_params={
  204. 'sites': '$site'
  205. }
  206. )
  207. site_group = DynamicModelChoiceField(
  208. label=_('Site group'),
  209. queryset=SiteGroup.objects.all(),
  210. required=False,
  211. initial_params={
  212. 'sites': '$site'
  213. }
  214. )
  215. site = DynamicModelChoiceField(
  216. label=_('Site'),
  217. queryset=Site.objects.all(),
  218. required=False,
  219. query_params={
  220. 'region_id': '$region',
  221. 'group_id': '$site_group',
  222. }
  223. )
  224. rack = DynamicModelChoiceField(
  225. label=_('Rack'),
  226. queryset=Rack.objects.all(),
  227. required=False,
  228. null_option='None',
  229. query_params={
  230. 'site_id': '$site'
  231. }
  232. )
  233. members = DynamicModelMultipleChoiceField(
  234. label=_('Members'),
  235. queryset=Device.objects.all(),
  236. required=False,
  237. query_params={
  238. 'virtual_chassis_id': 'null',
  239. 'site_id': '$site',
  240. 'rack_id': '$rack',
  241. }
  242. )
  243. initial_position = forms.IntegerField(
  244. label=_('Initial position'),
  245. initial=1,
  246. required=False,
  247. help_text=_('Position of the first member device. Increases by one for each additional member.')
  248. )
  249. fieldsets = (
  250. FieldSet('name', 'domain', 'description', 'tags', name=_('Virtual Chassis')),
  251. FieldSet('region', 'site_group', 'site', 'rack', 'members', 'initial_position', name=_('Member Devices')),
  252. )
  253. class Meta:
  254. model = VirtualChassis
  255. fields = [
  256. 'name', 'domain', 'description', 'region', 'site_group', 'site', 'rack', 'owner', 'members',
  257. 'initial_position', 'tags',
  258. ]
  259. def clean(self):
  260. super().clean()
  261. if self.cleaned_data['members'] and self.cleaned_data['initial_position'] is None:
  262. raise forms.ValidationError({
  263. 'initial_position': _("A position must be specified for the first VC member.")
  264. })
  265. def save(self, *args, **kwargs):
  266. instance = super().save(*args, **kwargs)
  267. # Assign VC members
  268. if instance.pk and self.cleaned_data['members']:
  269. initial_position = self.cleaned_data.get('initial_position', 1)
  270. for i, member in enumerate(self.cleaned_data['members'], start=initial_position):
  271. member.snapshot()
  272. member.virtual_chassis = instance
  273. member.vc_position = i
  274. member.save()
  275. return instance