| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529 |
- from django import forms
- from django.contrib.auth.models import User
- from django.contrib.contenttypes.models import ContentType
- from django.utils.safestring import mark_safe
- from django.utils.translation import gettext as _
- from dcim.models import DeviceRole, DeviceType, Platform, Region, Site, SiteGroup
- from tenancy.models import Tenant, TenantGroup
- from utilities.forms import (
- add_blank_choice, APISelectMultiple, BootstrapMixin, BulkEditForm, BulkEditNullBooleanSelect, ColorSelect,
- CommentField, ContentTypeMultipleChoiceField, CSVModelForm, DateTimePicker, DynamicModelMultipleChoiceField,
- JSONField, SlugField, StaticSelect2, BOOLEAN_WITH_BLANK_CHOICES,
- )
- from virtualization.models import Cluster, ClusterGroup
- from .choices import *
- from .models import ConfigContext, CustomField, ImageAttachment, JournalEntry, ObjectChange, Tag
- from .utils import FeatureQuery
- #
- # Custom fields
- #
- class CustomFieldForm(forms.Form):
- """
- Extend Form to include custom field support.
- """
- model = None
- def __init__(self, *args, **kwargs):
- if self.model is None:
- raise NotImplementedError("CustomFieldForm must specify a model class.")
- self.custom_fields = []
- super().__init__(*args, **kwargs)
- # Append relevant custom fields to the form instance
- obj_type = ContentType.objects.get_for_model(self.model)
- for cf in CustomField.objects.filter(content_types=obj_type):
- field_name = 'cf_{}'.format(cf.name)
- self.fields[field_name] = cf.to_form_field()
- # Annotate the field in the list of CustomField form fields
- self.custom_fields.append(field_name)
- class CustomFieldModelForm(forms.ModelForm):
- """
- Extend ModelForm to include custom field support.
- """
- def __init__(self, *args, **kwargs):
- self.obj_type = ContentType.objects.get_for_model(self._meta.model)
- self.custom_fields = []
- super().__init__(*args, **kwargs)
- self._append_customfield_fields()
- def _append_customfield_fields(self):
- """
- Append form fields for all CustomFields assigned to this model.
- """
- # Append form fields; assign initial values if modifying and existing object
- for cf in CustomField.objects.filter(content_types=self.obj_type):
- field_name = 'cf_{}'.format(cf.name)
- if self.instance.pk:
- self.fields[field_name] = cf.to_form_field(set_initial=False)
- self.fields[field_name].initial = self.instance.custom_field_data.get(cf.name)
- else:
- self.fields[field_name] = cf.to_form_field()
- # Annotate the field in the list of CustomField form fields
- self.custom_fields.append(field_name)
- def clean(self):
- # Save custom field data on instance
- for cf_name in self.custom_fields:
- self.instance.custom_field_data[cf_name[3:]] = self.cleaned_data.get(cf_name)
- return super().clean()
- class CustomFieldModelCSVForm(CSVModelForm, CustomFieldModelForm):
- def _append_customfield_fields(self):
- # Append form fields
- for cf in CustomField.objects.filter(content_types=self.obj_type):
- field_name = 'cf_{}'.format(cf.name)
- self.fields[field_name] = cf.to_form_field(for_csv_import=True)
- # Annotate the field in the list of CustomField form fields
- self.custom_fields.append(field_name)
- class CustomFieldBulkEditForm(BulkEditForm):
- def __init__(self, *args, **kwargs):
- super().__init__(*args, **kwargs)
- self.custom_fields = []
- self.obj_type = ContentType.objects.get_for_model(self.model)
- # Add all applicable CustomFields to the form
- custom_fields = CustomField.objects.filter(content_types=self.obj_type)
- for cf in custom_fields:
- # Annotate non-required custom fields as nullable
- if not cf.required:
- self.nullable_fields.append(cf.name)
- self.fields[cf.name] = cf.to_form_field(set_initial=False, enforce_required=False)
- # Annotate this as a custom field
- self.custom_fields.append(cf.name)
- class CustomFieldFilterForm(forms.Form):
- def __init__(self, *args, **kwargs):
- self.obj_type = ContentType.objects.get_for_model(self.model)
- super().__init__(*args, **kwargs)
- # Add all applicable CustomFields to the form
- custom_fields = CustomField.objects.filter(content_types=self.obj_type).exclude(
- filter_logic=CustomFieldFilterLogicChoices.FILTER_DISABLED
- )
- for cf in custom_fields:
- field_name = 'cf_{}'.format(cf.name)
- self.fields[field_name] = cf.to_form_field(set_initial=True, enforce_required=False)
- #
- # Tags
- #
- class TagForm(BootstrapMixin, forms.ModelForm):
- slug = SlugField()
- class Meta:
- model = Tag
- fields = [
- 'name', 'slug', 'color', 'description'
- ]
- fieldsets = (
- ('Tag', ('name', 'slug', 'color', 'description')),
- )
- class TagCSVForm(CSVModelForm):
- slug = SlugField()
- class Meta:
- model = Tag
- fields = Tag.csv_headers
- help_texts = {
- 'color': mark_safe('RGB color in hexadecimal (e.g. <code>00ff00</code>)'),
- }
- class AddRemoveTagsForm(forms.Form):
- def __init__(self, *args, **kwargs):
- super().__init__(*args, **kwargs)
- # Add add/remove tags fields
- self.fields['add_tags'] = DynamicModelMultipleChoiceField(
- queryset=Tag.objects.all(),
- required=False
- )
- self.fields['remove_tags'] = DynamicModelMultipleChoiceField(
- queryset=Tag.objects.all(),
- required=False
- )
- class TagFilterForm(BootstrapMixin, forms.Form):
- model = Tag
- q = forms.CharField(
- required=False,
- label=_('Search')
- )
- content_type_id = ContentTypeMultipleChoiceField(
- queryset=ContentType.objects.filter(FeatureQuery('tags').get_query()),
- required=False,
- label=_('Tagged object type')
- )
- class TagBulkEditForm(BootstrapMixin, BulkEditForm):
- pk = forms.ModelMultipleChoiceField(
- queryset=Tag.objects.all(),
- widget=forms.MultipleHiddenInput
- )
- color = forms.CharField(
- max_length=6,
- required=False,
- widget=ColorSelect()
- )
- description = forms.CharField(
- max_length=200,
- required=False
- )
- class Meta:
- nullable_fields = ['description']
- #
- # Config contexts
- #
- class ConfigContextForm(BootstrapMixin, forms.ModelForm):
- regions = DynamicModelMultipleChoiceField(
- queryset=Region.objects.all(),
- required=False
- )
- site_groups = DynamicModelMultipleChoiceField(
- queryset=SiteGroup.objects.all(),
- required=False
- )
- sites = DynamicModelMultipleChoiceField(
- queryset=Site.objects.all(),
- required=False
- )
- device_types = DynamicModelMultipleChoiceField(
- queryset=DeviceType.objects.all(),
- required=False
- )
- roles = DynamicModelMultipleChoiceField(
- queryset=DeviceRole.objects.all(),
- required=False
- )
- platforms = DynamicModelMultipleChoiceField(
- queryset=Platform.objects.all(),
- required=False
- )
- cluster_groups = DynamicModelMultipleChoiceField(
- queryset=ClusterGroup.objects.all(),
- required=False
- )
- clusters = DynamicModelMultipleChoiceField(
- queryset=Cluster.objects.all(),
- required=False
- )
- tenant_groups = DynamicModelMultipleChoiceField(
- queryset=TenantGroup.objects.all(),
- required=False
- )
- tenants = DynamicModelMultipleChoiceField(
- queryset=Tenant.objects.all(),
- required=False
- )
- tags = DynamicModelMultipleChoiceField(
- queryset=Tag.objects.all(),
- required=False
- )
- data = JSONField(
- label=''
- )
- class Meta:
- model = ConfigContext
- fields = (
- 'name', 'weight', 'description', 'is_active', 'regions', 'site_groups', 'sites', 'roles', 'device_types',
- 'platforms', 'cluster_groups', 'clusters', 'tenant_groups', 'tenants', 'tags', 'data',
- )
- class ConfigContextBulkEditForm(BootstrapMixin, BulkEditForm):
- pk = forms.ModelMultipleChoiceField(
- queryset=ConfigContext.objects.all(),
- widget=forms.MultipleHiddenInput
- )
- weight = forms.IntegerField(
- required=False,
- min_value=0
- )
- is_active = forms.NullBooleanField(
- required=False,
- widget=BulkEditNullBooleanSelect()
- )
- description = forms.CharField(
- required=False,
- max_length=100
- )
- class Meta:
- nullable_fields = [
- 'description',
- ]
- class ConfigContextFilterForm(BootstrapMixin, forms.Form):
- field_order = [
- 'q', 'region_id', 'site_group_id', 'site_id', 'role_id', 'platform_id', 'cluster_group_id', 'cluster_id',
- 'tenant_group_id', 'tenant_id',
- ]
- q = forms.CharField(
- required=False,
- label=_('Search')
- )
- region_id = DynamicModelMultipleChoiceField(
- queryset=Region.objects.all(),
- required=False,
- label=_('Regions')
- )
- site_group_id = DynamicModelMultipleChoiceField(
- queryset=SiteGroup.objects.all(),
- required=False,
- label=_('Site groups')
- )
- site_id = DynamicModelMultipleChoiceField(
- queryset=Site.objects.all(),
- required=False,
- label=_('Sites')
- )
- device_type_id = DynamicModelMultipleChoiceField(
- queryset=DeviceType.objects.all(),
- required=False,
- label=_('Device types')
- )
- role_id = DynamicModelMultipleChoiceField(
- queryset=DeviceRole.objects.all(),
- required=False,
- label=_('Roles')
- )
- platform_id = DynamicModelMultipleChoiceField(
- queryset=Platform.objects.all(),
- required=False,
- label=_('Platforms')
- )
- cluster_group_id = DynamicModelMultipleChoiceField(
- queryset=ClusterGroup.objects.all(),
- required=False,
- label=_('Cluster groups')
- )
- cluster_id = DynamicModelMultipleChoiceField(
- queryset=Cluster.objects.all(),
- required=False,
- label=_('Clusters')
- )
- tenant_group_id = DynamicModelMultipleChoiceField(
- queryset=TenantGroup.objects.all(),
- required=False,
- label=_('Tenant groups')
- )
- tenant_id = DynamicModelMultipleChoiceField(
- queryset=Tenant.objects.all(),
- required=False,
- label=_('Tenant')
- )
- tag = DynamicModelMultipleChoiceField(
- queryset=Tag.objects.all(),
- to_field_name='slug',
- required=False,
- label=_('Tags')
- )
- #
- # Filter form for local config context data
- #
- class LocalConfigContextFilterForm(forms.Form):
- local_context_data = forms.NullBooleanField(
- required=False,
- label=_('Has local config context data'),
- widget=StaticSelect2(
- choices=BOOLEAN_WITH_BLANK_CHOICES
- )
- )
- #
- # Image attachments
- #
- class ImageAttachmentForm(BootstrapMixin, forms.ModelForm):
- class Meta:
- model = ImageAttachment
- fields = [
- 'name', 'image',
- ]
- #
- # Journal entries
- #
- class JournalEntryForm(BootstrapMixin, forms.ModelForm):
- comments = CommentField()
- class Meta:
- model = JournalEntry
- fields = ['assigned_object_type', 'assigned_object_id', 'kind', 'comments']
- widgets = {
- 'assigned_object_type': forms.HiddenInput,
- 'assigned_object_id': forms.HiddenInput,
- }
- class JournalEntryBulkEditForm(BootstrapMixin, BulkEditForm):
- pk = forms.ModelMultipleChoiceField(
- queryset=JournalEntry.objects.all(),
- widget=forms.MultipleHiddenInput
- )
- kind = forms.ChoiceField(
- choices=JournalEntryKindChoices,
- required=False
- )
- comments = forms.CharField(
- required=False,
- widget=forms.Textarea()
- )
- class Meta:
- nullable_fields = []
- class JournalEntryFilterForm(BootstrapMixin, forms.Form):
- model = JournalEntry
- q = forms.CharField(
- required=False,
- label=_('Search')
- )
- created_after = forms.DateTimeField(
- required=False,
- label=_('After'),
- widget=DateTimePicker()
- )
- created_before = forms.DateTimeField(
- required=False,
- label=_('Before'),
- widget=DateTimePicker()
- )
- created_by_id = DynamicModelMultipleChoiceField(
- queryset=User.objects.all(),
- required=False,
- label=_('User'),
- widget=APISelectMultiple(
- api_url='/api/users/users/',
- )
- )
- assigned_object_type_id = DynamicModelMultipleChoiceField(
- queryset=ContentType.objects.all(),
- required=False,
- label=_('Object Type'),
- widget=APISelectMultiple(
- api_url='/api/extras/content-types/',
- )
- )
- kind = forms.ChoiceField(
- choices=add_blank_choice(JournalEntryKindChoices),
- required=False,
- widget=StaticSelect2()
- )
- #
- # Change logging
- #
- class ObjectChangeFilterForm(BootstrapMixin, forms.Form):
- model = ObjectChange
- q = forms.CharField(
- required=False,
- label=_('Search')
- )
- time_after = forms.DateTimeField(
- required=False,
- label=_('After'),
- widget=DateTimePicker()
- )
- time_before = forms.DateTimeField(
- required=False,
- label=_('Before'),
- widget=DateTimePicker()
- )
- action = forms.ChoiceField(
- choices=add_blank_choice(ObjectChangeActionChoices),
- required=False,
- widget=StaticSelect2()
- )
- user_id = DynamicModelMultipleChoiceField(
- queryset=User.objects.all(),
- required=False,
- label=_('User'),
- widget=APISelectMultiple(
- api_url='/api/users/users/',
- )
- )
- changed_object_type_id = DynamicModelMultipleChoiceField(
- queryset=ContentType.objects.all(),
- required=False,
- label=_('Object Type'),
- widget=APISelectMultiple(
- api_url='/api/extras/content-types/',
- )
- )
- #
- # Scripts
- #
- class ScriptForm(BootstrapMixin, forms.Form):
- _commit = forms.BooleanField(
- required=False,
- initial=True,
- label="Commit changes",
- help_text="Commit changes to the database (uncheck for a dry-run)"
- )
- def __init__(self, *args, **kwargs):
- super().__init__(*args, **kwargs)
- # Move _commit to the end of the form
- commit = self.fields.pop('_commit')
- self.fields['_commit'] = commit
- @property
- def requires_input(self):
- """
- A boolean indicating whether the form requires user input (ignore the _commit field).
- """
- return bool(len(self.fields) > 1)
|