Przeglądaj źródła

Virtulization Select2 forms

John Anderson 7 lat temu
rodzic
commit
6dcd48fef1
3 zmienionych plików z 163 dodań i 70 usunięć
  1. 1 4
      netbox/secrets/forms.py
  2. 5 3
      netbox/utilities/forms.py
  3. 157 63
      netbox/virtualization/forms.py

+ 1 - 4
netbox/secrets/forms.py

@@ -1,7 +1,6 @@
 from Crypto.Cipher import PKCS1_OAEP
 from Crypto.PublicKey import RSA
 from django import forms
-from django.db.models import Count
 from taggit.forms import TagField
 
 from dcim.models import Device
@@ -178,9 +177,7 @@ class SecretFilterForm(BootstrapMixin, CustomFieldFilterForm):
         label='Search'
     )
     role = FilterChoiceField(
-        queryset=SecretRole.objects.annotate(
-            filter_count=Count('secrets')
-        ),
+        queryset=SecretRole.objects.all(),
         to_field_name='slug',
         widget=APISelectMultiple(
             api_url="/api/secrets/secret-roles/",

+ 5 - 3
netbox/utilities/forms.py

@@ -717,15 +717,17 @@ class ChainedFieldsMixin(forms.BaseForm):
 
                 filters_dict = {}
                 for (db_field, parent_field) in field.chains:
-                    if self.is_bound and parent_field in self.data:
+                    if self.fields[parent_field].widget.attrs.get('nullable'):
+                        filters_dict[db_field] = None
+                    elif self.is_bound and parent_field in self.data and self.data[parent_field]:
                         filters_dict[db_field] = self.data[parent_field] or None
                     elif self.initial.get(parent_field):
                         filters_dict[db_field] = self.initial[parent_field]
-                    elif self.fields[parent_field].widget.attrs.get('nullable'):
-                        filters_dict[db_field] = None
                     else:
                         break
 
+                print(filters_dict)
+
                 if filters_dict:
                     field.queryset = field.queryset.filter(**filters_dict)
                 elif not self.is_bound and getattr(self, 'instance', None) and hasattr(self.instance, field_name):

+ 157 - 63
netbox/virtualization/forms.py

@@ -1,7 +1,5 @@
 from django import forms
 from django.core.exceptions import ValidationError
-from django.db.models import Count
-from mptt.forms import TreeNodeChoiceField
 from taggit.forms import TagField
 
 from dcim.constants import IFACE_FF_VIRTUAL, IFACE_MODE_ACCESS, IFACE_MODE_TAGGED_ALL
@@ -12,10 +10,10 @@ from ipam.models import IPAddress
 from tenancy.forms import TenancyForm
 from tenancy.models import Tenant
 from utilities.forms import (
-    AnnotatedMultipleChoiceField, APISelect, APISelectMultiple, BootstrapMixin, BulkEditForm, BulkEditNullBooleanSelect,
+    add_blank_choice, APISelect, APISelectMultiple, BootstrapMixin, BulkEditForm, BulkEditNullBooleanSelect,
     ChainedFieldsMixin, ChainedModelChoiceField, ChainedModelMultipleChoiceField, CommentField, ComponentForm,
-    ConfirmationForm, CSVChoiceField, ExpandableNameField, FilterChoiceField, FilterTreeNodeMultipleChoiceField,
-    JSONField, SlugField, SmallTextarea, add_blank_choice,
+    ConfirmationForm, CSVChoiceField, ExpandableNameField, FilterChoiceField, JSONField, SlugField,
+    SmallTextarea, StaticSelect2, StaticSelect2Multiple
 )
 from .constants import VM_STATUS_CHOICES
 from .models import Cluster, ClusterGroup, ClusterType, VirtualMachine
@@ -92,6 +90,17 @@ class ClusterForm(BootstrapMixin, CustomFieldForm):
         fields = [
             'name', 'type', 'group', 'site', 'comments', 'tags',
         ]
+        widgets = {
+            'type': APISelect(
+                api_url="/api/virtualization/cluster-types/"
+            ),
+            'group': APISelect(
+                api_url="/api/virtualization/cluster-groups/"
+            ),
+            'site': APISelect(
+                api_url="/api/dcim/sites/"
+            ),
+        }
 
 
 class ClusterCSVForm(forms.ModelForm):
@@ -134,15 +143,24 @@ class ClusterBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldBulkEdit
     )
     type = forms.ModelChoiceField(
         queryset=ClusterType.objects.all(),
-        required=False
+        required=False,
+        widget=APISelect(
+            api_url="/api/virtualization/cluster-types/"
+        )
     )
     group = forms.ModelChoiceField(
         queryset=ClusterGroup.objects.all(),
-        required=False
+        required=False,
+        widget=APISelect(
+            api_url="/api/virtualization/cluster-groups/"
+        )
     )
     site = forms.ModelChoiceField(
         queryset=Site.objects.all(),
-        required=False
+        required=False,
+        widget=APISelect(
+            api_url="/api/dcim/sites/"
+        )
     )
     comments = CommentField(
         widget=SmallTextarea()
@@ -158,37 +176,48 @@ class ClusterFilterForm(BootstrapMixin, CustomFieldFilterForm):
     model = Cluster
     q = forms.CharField(required=False, label='Search')
     type = FilterChoiceField(
-        queryset=ClusterType.objects.annotate(
-            filter_count=Count('clusters')
-        ),
+        queryset=ClusterType.objects.all(),
         to_field_name='slug',
         required=False,
+        widget=APISelectMultiple(
+            api_url="/api/virtualization/cluster-types/",
+            value_field='slug',
+        )
     )
     group = FilterChoiceField(
-        queryset=ClusterGroup.objects.annotate(
-            filter_count=Count('clusters')
-        ),
+        queryset=ClusterGroup.objects.all(),
         to_field_name='slug',
         null_label='-- None --',
         required=False,
+        widget=APISelectMultiple(
+            api_url="/api/virtualization/cluster-groups/",
+            value_field='slug',
+            null_option=True,
+        )
     )
     site = FilterChoiceField(
-        queryset=Site.objects.annotate(
-            filter_count=Count('clusters')
-        ),
+        queryset=Site.objects.all(),
         to_field_name='slug',
         null_label='-- None --',
         required=False,
+        widget=APISelectMultiple(
+            api_url="/api/dcim/sites/",
+            value_field='slug',
+            null_option=True,
+        )
     )
 
 
 class ClusterAddDevicesForm(BootstrapMixin, ChainedFieldsMixin, forms.Form):
-    region = TreeNodeChoiceField(
+    region = forms.ModelChoiceField(
         queryset=Region.objects.all(),
         required=False,
-        widget=forms.Select(
+        widget=APISelect(
+            api_url="/api/dcim/regions/",
+            filter_for={
+                "site": "region_id",
+            },
             attrs={
-                'filter-for': 'site',
                 'nullable': 'true',
             }
         )
@@ -200,9 +229,10 @@ class ClusterAddDevicesForm(BootstrapMixin, ChainedFieldsMixin, forms.Form):
         ),
         required=False,
         widget=APISelect(
-            api_url='/api/dcim/sites/?region_id={{region}}',
-            attrs={
-                'filter-for': 'rack',
+            api_url='/api/dcim/sites/',
+            filter_for={
+                "rack": "site_id",
+                "devices": "site_id",
             }
         )
     )
@@ -213,9 +243,11 @@ class ClusterAddDevicesForm(BootstrapMixin, ChainedFieldsMixin, forms.Form):
         ),
         required=False,
         widget=APISelect(
-            api_url='/api/dcim/racks/?site_id={{site}}',
+            api_url='/api/dcim/racks/',
+            filter_for={
+                "devices": "rack_id"
+            },
             attrs={
-                'filter-for': 'devices',
                 'nullable': 'true',
             }
         )
@@ -227,7 +259,7 @@ class ClusterAddDevicesForm(BootstrapMixin, ChainedFieldsMixin, forms.Form):
             ('rack', 'rack'),
         ),
         widget=APISelectMultiple(
-            api_url='/api/dcim/devices/?site_id={{site}}&rack_id={{rack}}',
+            api_url='/api/dcim/devices/',
             display_field='display_name',
             disabled_indicator='cluster'
         )
@@ -275,9 +307,12 @@ class VirtualMachineForm(BootstrapMixin, TenancyForm, CustomFieldForm):
     cluster_group = forms.ModelChoiceField(
         queryset=ClusterGroup.objects.all(),
         required=False,
-        widget=forms.Select(
+        widget=APISelect(
+            api_url='/api/virtualization/cluster-groups/',
+            filter_for={
+                "cluster": "group_id",
+            },
             attrs={
-                'filter-for': 'cluster',
                 'nullable': 'true',
             }
         )
@@ -288,7 +323,7 @@ class VirtualMachineForm(BootstrapMixin, TenancyForm, CustomFieldForm):
             ('group', 'cluster_group'),
         ),
         widget=APISelect(
-            api_url='/api/virtualization/clusters/?group_id={{cluster_group}}'
+            api_url='/api/virtualization/clusters/'
         )
     )
     tags = TagField(
@@ -308,6 +343,20 @@ class VirtualMachineForm(BootstrapMixin, TenancyForm, CustomFieldForm):
             'local_context_data': "Local config context data overwrites all sources contexts in the final rendered "
                                   "config context",
         }
+        widgets = {
+            "status": StaticSelect2(),
+            "role": APISelect(
+                api_url="/api/dcim/device-roles/",
+                additional_query_params={
+                    "vm_role": "true"
+                }
+            ),
+            'primary_ip4': StaticSelect2(),
+            'primary_ip6': StaticSelect2(),
+            'platform': APISelect(
+                api_url='/api/dcim/platforms/'
+            )
+        }
 
     def __init__(self, *args, **kwargs):
 
@@ -413,25 +462,41 @@ class VirtualMachineBulkEditForm(BootstrapMixin, AddRemoveTagsForm, CustomFieldB
     status = forms.ChoiceField(
         choices=add_blank_choice(VM_STATUS_CHOICES),
         required=False,
-        initial=''
+        initial='',
+        widget=StaticSelect2(),
     )
     cluster = forms.ModelChoiceField(
         queryset=Cluster.objects.all(),
-        required=False
+        required=False,
+        widget=APISelect(
+            api_url='/api/virtualization/clusters/'
+        )
     )
     role = forms.ModelChoiceField(
         queryset=DeviceRole.objects.filter(
             vm_role=True
         ),
-        required=False
+        required=False,
+        widget=APISelect(
+            api_url="/api/dcim/device-roles/",
+            additional_query_params={
+                "vm_role": "true"
+            }
+        )
     )
     tenant = forms.ModelChoiceField(
         queryset=Tenant.objects.all(),
-        required=False
+        required=False,
+        widget=APISelect(
+            api_url='/api/tenancy/tenants/'
+        )
     )
     platform = forms.ModelChoiceField(
         queryset=Platform.objects.all(),
-        required=False
+        required=False,
+        widget=APISelect(
+            api_url='/api/dcim/platforms/'
+        )
     )
     vcpus = forms.IntegerField(
         required=False,
@@ -464,59 +529,87 @@ class VirtualMachineFilterForm(BootstrapMixin, CustomFieldFilterForm):
     cluster_group = FilterChoiceField(
         queryset=ClusterGroup.objects.all(),
         to_field_name='slug',
-        null_label='-- None --'
+        null_label='-- None --',
+        widget=APISelectMultiple(
+            api_url='/api/virtualization/cluster-groups/',
+            value_field="slug",
+            null_option=True,
+        )
     )
     cluster_type = FilterChoiceField(
         queryset=ClusterType.objects.all(),
         to_field_name='slug',
-        null_label='-- None --'
+        null_label='-- None --',
+        widget=APISelectMultiple(
+            api_url='/api/virtualization/cluster-types/',
+            value_field="slug",
+            null_option=True,
+        )
     )
     cluster_id = FilterChoiceField(
-        queryset=Cluster.objects.annotate(
-            filter_count=Count('virtual_machines')
-        ),
-        label='Cluster'
+        queryset=Cluster.objects.all(),
+        label='Cluster',
+        widget=APISelectMultiple(
+            api_url='/api/virtualization/clusters/',
+        )
     )
-    region = FilterTreeNodeMultipleChoiceField(
+    region = FilterChoiceField(
         queryset=Region.objects.all(),
         to_field_name='slug',
         required=False,
+        widget=APISelectMultiple(
+            api_url='/api/dcim/regions/',
+            value_field="slug",
+            null_option=True,
+        )
     )
     site = FilterChoiceField(
-        queryset=Site.objects.annotate(
-            filter_count=Count('clusters__virtual_machines')
-        ),
+        queryset=Site.objects.all(),
         to_field_name='slug',
-        null_label='-- None --'
+        null_label='-- None --',
+        widget=APISelectMultiple(
+            api_url='/api/dcim/sites/',
+            value_field="slug",
+            null_option=True,
+        )
     )
     role = FilterChoiceField(
-        queryset=DeviceRole.objects.filter(
-            vm_role=True
-        ).annotate(
-            filter_count=Count('virtual_machines')
-        ),
+        queryset=DeviceRole.objects.filter(vm_role=True),
         to_field_name='slug',
-        null_label='-- None --'
+        null_label='-- None --',
+        widget=APISelectMultiple(
+            api_url='/api/dcim/device-roles/',
+            value_field="slug",
+            null_option=True,
+            additional_query_params={
+                'vm_role': 'true'
+            }
+        )
     )
-    status = AnnotatedMultipleChoiceField(
+    status = forms.MultipleChoiceField(
         choices=VM_STATUS_CHOICES,
-        annotate=VirtualMachine.objects.all(),
-        annotate_field='status',
-        required=False
+        required=False,
+        widget=StaticSelect2Multiple()
     )
     tenant = FilterChoiceField(
-        queryset=Tenant.objects.annotate(
-            filter_count=Count('virtual_machines')
-        ),
+        queryset=Tenant.objects.all(),
         to_field_name='slug',
-        null_label='-- None --'
+        null_label='-- None --',
+        widget=APISelectMultiple(
+            api_url='/api/tenancy/tenants/',
+            value_field="slug",
+            null_option=True,
+        )
     )
     platform = FilterChoiceField(
-        queryset=Platform.objects.annotate(
-            filter_count=Count('virtual_machines')
-        ),
+        queryset=Platform.objects.all(),
         to_field_name='slug',
-        null_label='-- None --'
+        null_label='-- None --',
+        widget=APISelectMultiple(
+            api_url='/api/dcim/platforms/',
+            value_field="slug",
+            null_option=True,
+        )
     )
 
 
@@ -538,6 +631,7 @@ class InterfaceForm(BootstrapMixin, forms.ModelForm):
         widgets = {
             'virtual_machine': forms.HiddenInput(),
             'form_factor': forms.HiddenInput(),
+            'mode': StaticSelect2()
         }
         labels = {
             'mode': '802.1Q Mode',