forms.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451
  1. from django import forms
  2. from django.contrib.auth.models import User
  3. from django.contrib.contenttypes.models import ContentType
  4. from mptt.forms import TreeNodeMultipleChoiceField
  5. from taggit.forms import TagField as TagField_
  6. from dcim.models import DeviceRole, Platform, Region, Site
  7. from tenancy.models import Tenant, TenantGroup
  8. from utilities.forms import (
  9. add_blank_choice, APISelectMultiple, BootstrapMixin, BulkEditForm, BulkEditNullBooleanSelect, ColorSelect,
  10. ContentTypeSelect, CSVModelForm, DateTimePicker, DynamicModelMultipleChoiceField, JSONField, SlugField,
  11. StaticSelect2, StaticSelect2Multiple, BOOLEAN_WITH_BLANK_CHOICES,
  12. )
  13. from virtualization.models import Cluster, ClusterGroup
  14. from .choices import *
  15. from .models import ConfigContext, CustomField, CustomFieldValue, ImageAttachment, ObjectChange, Tag
  16. #
  17. # Custom fields
  18. #
  19. class CustomFieldModelForm(forms.ModelForm):
  20. def __init__(self, *args, **kwargs):
  21. self.obj_type = ContentType.objects.get_for_model(self._meta.model)
  22. self.custom_fields = []
  23. self.custom_field_values = {}
  24. super().__init__(*args, **kwargs)
  25. self._append_customfield_fields()
  26. def _append_customfield_fields(self):
  27. """
  28. Append form fields for all CustomFields assigned to this model.
  29. """
  30. # Retrieve initial CustomField values for the instance
  31. if self.instance.pk:
  32. for cfv in CustomFieldValue.objects.filter(
  33. obj_type=self.obj_type,
  34. obj_id=self.instance.pk
  35. ).prefetch_related('field'):
  36. self.custom_field_values[cfv.field.name] = cfv.serialized_value
  37. # Append form fields; assign initial values if modifying and existing object
  38. for cf in CustomField.objects.filter(obj_type=self.obj_type):
  39. field_name = 'cf_{}'.format(cf.name)
  40. if self.instance.pk:
  41. self.fields[field_name] = cf.to_form_field(set_initial=False)
  42. self.fields[field_name].initial = self.custom_field_values.get(cf.name)
  43. else:
  44. self.fields[field_name] = cf.to_form_field()
  45. # Annotate the field in the list of CustomField form fields
  46. self.custom_fields.append(field_name)
  47. def _save_custom_fields(self):
  48. for field_name in self.custom_fields:
  49. try:
  50. cfv = CustomFieldValue.objects.prefetch_related('field').get(
  51. field=self.fields[field_name].model,
  52. obj_type=self.obj_type,
  53. obj_id=self.instance.pk
  54. )
  55. except CustomFieldValue.DoesNotExist:
  56. # Skip this field if none exists already and its value is empty
  57. if self.cleaned_data[field_name] in [None, '']:
  58. continue
  59. cfv = CustomFieldValue(
  60. field=self.fields[field_name].model,
  61. obj_type=self.obj_type,
  62. obj_id=self.instance.pk
  63. )
  64. cfv.value = self.cleaned_data[field_name]
  65. cfv.save()
  66. def save(self, commit=True):
  67. obj = super().save(commit)
  68. # Handle custom fields the same way we do M2M fields
  69. if commit:
  70. self._save_custom_fields()
  71. else:
  72. self.save_custom_fields = self._save_custom_fields
  73. return obj
  74. class CustomFieldModelCSVForm(CSVModelForm, CustomFieldModelForm):
  75. def _append_customfield_fields(self):
  76. # Append form fields
  77. for cf in CustomField.objects.filter(obj_type=self.obj_type):
  78. field_name = 'cf_{}'.format(cf.name)
  79. self.fields[field_name] = cf.to_form_field(for_csv_import=True)
  80. # Annotate the field in the list of CustomField form fields
  81. self.custom_fields.append(field_name)
  82. class CustomFieldBulkEditForm(BulkEditForm):
  83. def __init__(self, *args, **kwargs):
  84. super().__init__(*args, **kwargs)
  85. self.custom_fields = []
  86. self.obj_type = ContentType.objects.get_for_model(self.model)
  87. # Add all applicable CustomFields to the form
  88. custom_fields = CustomField.objects.filter(obj_type=self.obj_type)
  89. for cf in custom_fields:
  90. # Annotate non-required custom fields as nullable
  91. if not cf.required:
  92. self.nullable_fields.append(cf.name)
  93. self.fields[cf.name] = cf.to_form_field(set_initial=False, enforce_required=False)
  94. # Annotate this as a custom field
  95. self.custom_fields.append(cf.name)
  96. class CustomFieldFilterForm(forms.Form):
  97. def __init__(self, *args, **kwargs):
  98. self.obj_type = ContentType.objects.get_for_model(self.model)
  99. super().__init__(*args, **kwargs)
  100. # Add all applicable CustomFields to the form
  101. custom_fields = CustomField.objects.filter(obj_type=self.obj_type).exclude(
  102. filter_logic=CustomFieldFilterLogicChoices.FILTER_DISABLED
  103. )
  104. for cf in custom_fields:
  105. field_name = 'cf_{}'.format(cf.name)
  106. self.fields[field_name] = cf.to_form_field(set_initial=True, enforce_required=False)
  107. #
  108. # Tags
  109. #
  110. class TagField(TagField_):
  111. def widget_attrs(self, widget):
  112. # Apply the "tagfield" CSS class to trigger the special API-based selection widget for tags
  113. return {
  114. 'class': 'tagfield'
  115. }
  116. class TagForm(BootstrapMixin, forms.ModelForm):
  117. slug = SlugField()
  118. class Meta:
  119. model = Tag
  120. fields = [
  121. 'name', 'slug', 'color', 'description'
  122. ]
  123. class AddRemoveTagsForm(forms.Form):
  124. def __init__(self, *args, **kwargs):
  125. super().__init__(*args, **kwargs)
  126. # Add add/remove tags fields
  127. self.fields['add_tags'] = DynamicModelMultipleChoiceField(
  128. queryset=Tag.objects.all(),
  129. required=False
  130. )
  131. self.fields['remove_tags'] = DynamicModelMultipleChoiceField(
  132. queryset=Tag.objects.all(),
  133. required=False
  134. )
  135. class TagFilterForm(BootstrapMixin, forms.Form):
  136. model = Tag
  137. q = forms.CharField(
  138. required=False,
  139. label='Search'
  140. )
  141. class TagBulkEditForm(BootstrapMixin, BulkEditForm):
  142. pk = forms.ModelMultipleChoiceField(
  143. queryset=Tag.objects.all(),
  144. widget=forms.MultipleHiddenInput
  145. )
  146. color = forms.CharField(
  147. max_length=6,
  148. required=False,
  149. widget=ColorSelect()
  150. )
  151. description = forms.CharField(
  152. max_length=200,
  153. required=False
  154. )
  155. class Meta:
  156. nullable_fields = ['description']
  157. #
  158. # Config contexts
  159. #
  160. class ConfigContextForm(BootstrapMixin, forms.ModelForm):
  161. regions = TreeNodeMultipleChoiceField(
  162. queryset=Region.objects.all(),
  163. required=False,
  164. widget=StaticSelect2Multiple()
  165. )
  166. sites = DynamicModelMultipleChoiceField(
  167. queryset=Site.objects.all(),
  168. required=False
  169. )
  170. roles = DynamicModelMultipleChoiceField(
  171. queryset=DeviceRole.objects.all(),
  172. required=False
  173. )
  174. platforms = DynamicModelMultipleChoiceField(
  175. queryset=Platform.objects.all(),
  176. required=False
  177. )
  178. cluster_groups = DynamicModelMultipleChoiceField(
  179. queryset=ClusterGroup.objects.all(),
  180. required=False
  181. )
  182. clusters = DynamicModelMultipleChoiceField(
  183. queryset=Cluster.objects.all(),
  184. required=False
  185. )
  186. tenant_groups = DynamicModelMultipleChoiceField(
  187. queryset=TenantGroup.objects.all(),
  188. required=False
  189. )
  190. tenants = DynamicModelMultipleChoiceField(
  191. queryset=Tenant.objects.all(),
  192. required=False
  193. )
  194. tags = DynamicModelMultipleChoiceField(
  195. queryset=Tag.objects.all(),
  196. required=False
  197. )
  198. data = JSONField(
  199. label=''
  200. )
  201. class Meta:
  202. model = ConfigContext
  203. fields = (
  204. 'name', 'weight', 'description', 'is_active', 'regions', 'sites', 'roles', 'platforms', 'cluster_groups',
  205. 'clusters', 'tenant_groups', 'tenants', 'tags', 'data',
  206. )
  207. class ConfigContextBulkEditForm(BootstrapMixin, BulkEditForm):
  208. pk = forms.ModelMultipleChoiceField(
  209. queryset=ConfigContext.objects.all(),
  210. widget=forms.MultipleHiddenInput
  211. )
  212. weight = forms.IntegerField(
  213. required=False,
  214. min_value=0
  215. )
  216. is_active = forms.NullBooleanField(
  217. required=False,
  218. widget=BulkEditNullBooleanSelect()
  219. )
  220. description = forms.CharField(
  221. required=False,
  222. max_length=100
  223. )
  224. class Meta:
  225. nullable_fields = [
  226. 'description',
  227. ]
  228. class ConfigContextFilterForm(BootstrapMixin, forms.Form):
  229. q = forms.CharField(
  230. required=False,
  231. label='Search'
  232. )
  233. region = DynamicModelMultipleChoiceField(
  234. queryset=Region.objects.all(),
  235. to_field_name='slug',
  236. required=False,
  237. widget=APISelectMultiple(
  238. value_field="slug",
  239. )
  240. )
  241. site = DynamicModelMultipleChoiceField(
  242. queryset=Site.objects.all(),
  243. to_field_name='slug',
  244. required=False,
  245. widget=APISelectMultiple(
  246. value_field="slug",
  247. )
  248. )
  249. role = DynamicModelMultipleChoiceField(
  250. queryset=DeviceRole.objects.all(),
  251. to_field_name='slug',
  252. required=False,
  253. widget=APISelectMultiple(
  254. value_field="slug",
  255. )
  256. )
  257. platform = DynamicModelMultipleChoiceField(
  258. queryset=Platform.objects.all(),
  259. to_field_name='slug',
  260. required=False,
  261. widget=APISelectMultiple(
  262. value_field="slug",
  263. )
  264. )
  265. cluster_group = DynamicModelMultipleChoiceField(
  266. queryset=ClusterGroup.objects.all(),
  267. to_field_name='slug',
  268. required=False,
  269. widget=APISelectMultiple(
  270. value_field="slug",
  271. )
  272. )
  273. cluster_id = DynamicModelMultipleChoiceField(
  274. queryset=Cluster.objects.all(),
  275. required=False,
  276. label='Cluster'
  277. )
  278. tenant_group = DynamicModelMultipleChoiceField(
  279. queryset=TenantGroup.objects.all(),
  280. to_field_name='slug',
  281. required=False,
  282. widget=APISelectMultiple(
  283. value_field="slug",
  284. )
  285. )
  286. tenant = DynamicModelMultipleChoiceField(
  287. queryset=Tenant.objects.all(),
  288. to_field_name='slug',
  289. required=False,
  290. widget=APISelectMultiple(
  291. value_field="slug",
  292. )
  293. )
  294. tag = DynamicModelMultipleChoiceField(
  295. queryset=Tag.objects.all(),
  296. to_field_name='slug',
  297. required=False,
  298. widget=APISelectMultiple(
  299. value_field="slug",
  300. )
  301. )
  302. #
  303. # Filter form for local config context data
  304. #
  305. class LocalConfigContextFilterForm(forms.Form):
  306. local_context_data = forms.NullBooleanField(
  307. required=False,
  308. label='Has local config context data',
  309. widget=StaticSelect2(
  310. choices=BOOLEAN_WITH_BLANK_CHOICES
  311. )
  312. )
  313. #
  314. # Image attachments
  315. #
  316. class ImageAttachmentForm(BootstrapMixin, forms.ModelForm):
  317. class Meta:
  318. model = ImageAttachment
  319. fields = [
  320. 'name', 'image',
  321. ]
  322. #
  323. # Change logging
  324. #
  325. class ObjectChangeFilterForm(BootstrapMixin, forms.Form):
  326. model = ObjectChange
  327. q = forms.CharField(
  328. required=False,
  329. label='Search'
  330. )
  331. time_after = forms.DateTimeField(
  332. label='After',
  333. required=False,
  334. widget=DateTimePicker()
  335. )
  336. time_before = forms.DateTimeField(
  337. label='Before',
  338. required=False,
  339. widget=DateTimePicker()
  340. )
  341. action = forms.ChoiceField(
  342. choices=add_blank_choice(ObjectChangeActionChoices),
  343. required=False,
  344. widget=StaticSelect2()
  345. )
  346. # TODO: Convert to DynamicModelMultipleChoiceField once we have an API endpoint for users
  347. user = forms.ModelChoiceField(
  348. queryset=User.objects.order_by('username'),
  349. required=False,
  350. widget=StaticSelect2()
  351. )
  352. changed_object_type = forms.ModelChoiceField(
  353. queryset=ContentType.objects.order_by('model'),
  354. required=False,
  355. widget=ContentTypeSelect(),
  356. label='Object Type'
  357. )
  358. #
  359. # Scripts
  360. #
  361. class ScriptForm(BootstrapMixin, forms.Form):
  362. _commit = forms.BooleanField(
  363. required=False,
  364. initial=True,
  365. label="Commit changes",
  366. help_text="Commit changes to the database (uncheck for a dry-run)"
  367. )
  368. def __init__(self, *args, **kwargs):
  369. super().__init__(*args, **kwargs)
  370. # Move _commit to the end of the form
  371. commit = self.fields.pop('_commit')
  372. self.fields['_commit'] = commit
  373. @property
  374. def requires_input(self):
  375. """
  376. A boolean indicating whether the form requires user input (ignore the _commit field).
  377. """
  378. return bool(len(self.fields) > 1)