object_create.py 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. from django import forms
  2. from dcim.models import *
  3. from netbox.forms import NetBoxModelForm
  4. from utilities.forms import (
  5. BootstrapMixin, DynamicModelChoiceField, DynamicModelMultipleChoiceField, ExpandableNameField,
  6. )
  7. __all__ = (
  8. 'ComponentTemplateCreateForm',
  9. 'DeviceComponentCreateForm',
  10. 'FrontPortCreateForm',
  11. 'FrontPortTemplateCreateForm',
  12. 'InventoryItemCreateForm',
  13. 'ModularComponentTemplateCreateForm',
  14. 'ModuleBayCreateForm',
  15. 'ModuleBayTemplateCreateForm',
  16. 'VirtualChassisCreateForm',
  17. )
  18. class ComponentCreateForm(BootstrapMixin, forms.Form):
  19. """
  20. Subclass this form when facilitating the creation of one or more device component or component templates based on
  21. a name pattern.
  22. """
  23. name_pattern = ExpandableNameField(
  24. label='Name'
  25. )
  26. label_pattern = ExpandableNameField(
  27. label='Label',
  28. required=False,
  29. help_text='Alphanumeric ranges are supported. (Must match the number of names being created.)'
  30. )
  31. def clean(self):
  32. super().clean()
  33. # Validate that all patterned fields generate an equal number of values
  34. patterned_fields = [
  35. field_name for field_name in self.fields if field_name.endswith('_pattern')
  36. ]
  37. pattern_count = len(self.cleaned_data['name_pattern'])
  38. for field_name in patterned_fields:
  39. value_count = len(self.cleaned_data[field_name])
  40. if self.cleaned_data[field_name] and value_count != pattern_count:
  41. raise forms.ValidationError({
  42. field_name: f'The provided pattern specifies {value_count} values, but {pattern_count} are '
  43. f'expected.'
  44. }, code='label_pattern_mismatch')
  45. class ComponentTemplateCreateForm(ComponentCreateForm):
  46. """
  47. Creation form for component templates that can be assigned only to a DeviceType.
  48. """
  49. device_type = DynamicModelChoiceField(
  50. queryset=DeviceType.objects.all(),
  51. )
  52. field_order = ('device_type', 'name_pattern', 'label_pattern')
  53. class ModularComponentTemplateCreateForm(ComponentCreateForm):
  54. """
  55. Creation form for component templates that can be assigned to either a DeviceType *or* a ModuleType.
  56. """
  57. device_type = DynamicModelChoiceField(
  58. queryset=DeviceType.objects.all(),
  59. required=False
  60. )
  61. module_type = DynamicModelChoiceField(
  62. queryset=ModuleType.objects.all(),
  63. required=False
  64. )
  65. field_order = ('device_type', 'module_type', 'name_pattern', 'label_pattern')
  66. class DeviceComponentCreateForm(ComponentCreateForm):
  67. device = DynamicModelChoiceField(
  68. queryset=Device.objects.all()
  69. )
  70. field_order = ('device', 'name_pattern', 'label_pattern')
  71. class FrontPortTemplateCreateForm(ModularComponentTemplateCreateForm):
  72. rear_port_set = forms.MultipleChoiceField(
  73. choices=[],
  74. label='Rear ports',
  75. help_text='Select one rear port assignment for each front port being created.',
  76. )
  77. field_order = (
  78. 'device_type', 'name_pattern', 'label_pattern', 'rear_port_set',
  79. )
  80. def __init__(self, *args, **kwargs):
  81. super().__init__(*args, **kwargs)
  82. # TODO: This needs better validation
  83. if 'device_type' in self.initial or self.data.get('device_type'):
  84. parent = DeviceType.objects.get(
  85. pk=self.initial.get('device_type') or self.data.get('device_type')
  86. )
  87. elif 'module_type' in self.initial or self.data.get('module_type'):
  88. parent = ModuleType.objects.get(
  89. pk=self.initial.get('module_type') or self.data.get('module_type')
  90. )
  91. else:
  92. return
  93. # Determine which rear port positions are occupied. These will be excluded from the list of available mappings.
  94. occupied_port_positions = [
  95. (front_port.rear_port_id, front_port.rear_port_position)
  96. for front_port in parent.frontporttemplates.all()
  97. ]
  98. # Populate rear port choices
  99. choices = []
  100. rear_ports = parent.rearporttemplates.all()
  101. for rear_port in rear_ports:
  102. for i in range(1, rear_port.positions + 1):
  103. if (rear_port.pk, i) not in occupied_port_positions:
  104. choices.append(
  105. ('{}:{}'.format(rear_port.pk, i), '{}:{}'.format(rear_port.name, i))
  106. )
  107. self.fields['rear_port_set'].choices = choices
  108. def get_iterative_data(self, iteration):
  109. # Assign rear port and position from selected set
  110. rear_port, position = self.cleaned_data['rear_port_set'][iteration].split(':')
  111. return {
  112. 'rear_port': int(rear_port),
  113. 'rear_port_position': int(position),
  114. }
  115. class FrontPortCreateForm(DeviceComponentCreateForm):
  116. rear_port_set = forms.MultipleChoiceField(
  117. choices=[],
  118. label='Rear ports',
  119. help_text='Select one rear port assignment for each front port being created.',
  120. )
  121. field_order = (
  122. 'device', 'name_pattern', 'label_pattern', 'rear_port_set',
  123. )
  124. def __init__(self, *args, **kwargs):
  125. super().__init__(*args, **kwargs)
  126. device = Device.objects.get(
  127. pk=self.initial.get('device') or self.data.get('device')
  128. )
  129. # Determine which rear port positions are occupied. These will be excluded from the list of available
  130. # mappings.
  131. occupied_port_positions = [
  132. (front_port.rear_port_id, front_port.rear_port_position)
  133. for front_port in device.frontports.all()
  134. ]
  135. # Populate rear port choices
  136. choices = []
  137. rear_ports = RearPort.objects.filter(device=device)
  138. for rear_port in rear_ports:
  139. for i in range(1, rear_port.positions + 1):
  140. if (rear_port.pk, i) not in occupied_port_positions:
  141. choices.append(
  142. ('{}:{}'.format(rear_port.pk, i), '{}:{}'.format(rear_port.name, i))
  143. )
  144. self.fields['rear_port_set'].choices = choices
  145. def get_iterative_data(self, iteration):
  146. # Assign rear port and position from selected set
  147. rear_port, position = self.cleaned_data['rear_port_set'][iteration].split(':')
  148. return {
  149. 'rear_port': int(rear_port),
  150. 'rear_port_position': int(position),
  151. }
  152. class ModuleBayTemplateCreateForm(ComponentTemplateCreateForm):
  153. position_pattern = ExpandableNameField(
  154. label='Position',
  155. required=False,
  156. help_text='Alphanumeric ranges are supported. (Must match the number of names being created.)'
  157. )
  158. field_order = ('device_type', 'name_pattern', 'label_pattern', 'position_pattern')
  159. class ModuleBayCreateForm(DeviceComponentCreateForm):
  160. position_pattern = ExpandableNameField(
  161. label='Position',
  162. required=False,
  163. help_text='Alphanumeric ranges are supported. (Must match the number of names being created.)'
  164. )
  165. field_order = ('device', 'name_pattern', 'label_pattern', 'position_pattern')
  166. class InventoryItemCreateForm(ComponentCreateForm):
  167. # Device is assigned by the model form
  168. field_order = ('name_pattern', 'label_pattern')
  169. class VirtualChassisCreateForm(NetBoxModelForm):
  170. region = DynamicModelChoiceField(
  171. queryset=Region.objects.all(),
  172. required=False,
  173. initial_params={
  174. 'sites': '$site'
  175. }
  176. )
  177. site_group = DynamicModelChoiceField(
  178. queryset=SiteGroup.objects.all(),
  179. required=False,
  180. initial_params={
  181. 'sites': '$site'
  182. }
  183. )
  184. site = DynamicModelChoiceField(
  185. queryset=Site.objects.all(),
  186. required=False,
  187. query_params={
  188. 'region_id': '$region',
  189. 'group_id': '$site_group',
  190. }
  191. )
  192. rack = DynamicModelChoiceField(
  193. queryset=Rack.objects.all(),
  194. required=False,
  195. null_option='None',
  196. query_params={
  197. 'site_id': '$site'
  198. }
  199. )
  200. members = DynamicModelMultipleChoiceField(
  201. queryset=Device.objects.all(),
  202. required=False,
  203. query_params={
  204. 'site_id': '$site',
  205. 'rack_id': '$rack',
  206. }
  207. )
  208. initial_position = forms.IntegerField(
  209. initial=1,
  210. required=False,
  211. help_text='Position of the first member device. Increases by one for each additional member.'
  212. )
  213. class Meta:
  214. model = VirtualChassis
  215. fields = [
  216. 'name', 'domain', 'region', 'site_group', 'site', 'rack', 'members', 'initial_position', 'tags',
  217. ]
  218. def clean(self):
  219. super().clean()
  220. if self.cleaned_data['members'] and self.cleaned_data['initial_position'] is None:
  221. raise forms.ValidationError({
  222. 'initial_position': "A position must be specified for the first VC member."
  223. })
  224. def save(self, *args, **kwargs):
  225. instance = super().save(*args, **kwargs)
  226. # Assign VC members
  227. if instance.pk and self.cleaned_data['members']:
  228. initial_position = self.cleaned_data.get('initial_position', 1)
  229. for i, member in enumerate(self.cleaned_data['members'], start=initial_position):
  230. member.virtual_chassis = instance
  231. member.vc_position = i
  232. member.save()
  233. return instance