|
|
@@ -1,13 +1,16 @@
|
|
|
import django_filters
|
|
|
+from django.contrib.contenttypes.models import ContentType
|
|
|
from django.db.models import Q
|
|
|
from django.utils.translation import gettext as _
|
|
|
|
|
|
from dcim.filtersets import CabledObjectFilterSet
|
|
|
-from dcim.models import Region, Site, SiteGroup
|
|
|
+from dcim.models import Interface, Location, Region, Site, SiteGroup
|
|
|
from ipam.models import ASN
|
|
|
from netbox.filtersets import NetBoxModelFilterSet, OrganizationalModelFilterSet
|
|
|
from tenancy.filtersets import ContactModelFilterSet, TenancyFilterSet
|
|
|
-from utilities.filters import TreeNodeMultipleChoiceFilter
|
|
|
+from utilities.filters import (
|
|
|
+ ContentTypeFilter, MultiValueCharFilter, MultiValueNumberFilter, TreeNodeMultipleChoiceFilter,
|
|
|
+)
|
|
|
from .choices import *
|
|
|
from .models import *
|
|
|
|
|
|
@@ -20,43 +23,46 @@ __all__ = (
|
|
|
'ProviderNetworkFilterSet',
|
|
|
'ProviderAccountFilterSet',
|
|
|
'ProviderFilterSet',
|
|
|
+ 'VirtualCircuitFilterSet',
|
|
|
+ 'VirtualCircuitTerminationFilterSet',
|
|
|
+ 'VirtualCircuitTypeFilterSet',
|
|
|
)
|
|
|
|
|
|
|
|
|
class ProviderFilterSet(NetBoxModelFilterSet, ContactModelFilterSet):
|
|
|
region_id = TreeNodeMultipleChoiceFilter(
|
|
|
queryset=Region.objects.all(),
|
|
|
- field_name='circuits__terminations__site__region',
|
|
|
+ field_name='circuits__terminations___region',
|
|
|
lookup_expr='in',
|
|
|
label=_('Region (ID)'),
|
|
|
)
|
|
|
region = TreeNodeMultipleChoiceFilter(
|
|
|
queryset=Region.objects.all(),
|
|
|
- field_name='circuits__terminations__site__region',
|
|
|
+ field_name='circuits__terminations___region',
|
|
|
lookup_expr='in',
|
|
|
to_field_name='slug',
|
|
|
label=_('Region (slug)'),
|
|
|
)
|
|
|
site_group_id = TreeNodeMultipleChoiceFilter(
|
|
|
queryset=SiteGroup.objects.all(),
|
|
|
- field_name='circuits__terminations__site__group',
|
|
|
+ field_name='circuits__terminations___site_group',
|
|
|
lookup_expr='in',
|
|
|
label=_('Site group (ID)'),
|
|
|
)
|
|
|
site_group = TreeNodeMultipleChoiceFilter(
|
|
|
queryset=SiteGroup.objects.all(),
|
|
|
- field_name='circuits__terminations__site__group',
|
|
|
+ field_name='circuits__terminations___site_group',
|
|
|
lookup_expr='in',
|
|
|
to_field_name='slug',
|
|
|
label=_('Site group (slug)'),
|
|
|
)
|
|
|
site_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
- field_name='circuits__terminations__site',
|
|
|
+ field_name='circuits__terminations___site',
|
|
|
queryset=Site.objects.all(),
|
|
|
label=_('Site'),
|
|
|
)
|
|
|
site = django_filters.ModelMultipleChoiceFilter(
|
|
|
- field_name='circuits__terminations__site__slug',
|
|
|
+ field_name='circuits__terminations___site__slug',
|
|
|
queryset=Site.objects.all(),
|
|
|
to_field_name='slug',
|
|
|
label=_('Site (slug)'),
|
|
|
@@ -173,7 +179,7 @@ class CircuitFilterSet(NetBoxModelFilterSet, TenancyFilterSet, ContactModelFilte
|
|
|
label=_('Provider account (account)'),
|
|
|
)
|
|
|
provider_network_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
- field_name='terminations__provider_network',
|
|
|
+ field_name='terminations___provider_network',
|
|
|
queryset=ProviderNetwork.objects.all(),
|
|
|
label=_('Provider network (ID)'),
|
|
|
)
|
|
|
@@ -193,37 +199,37 @@ class CircuitFilterSet(NetBoxModelFilterSet, TenancyFilterSet, ContactModelFilte
|
|
|
)
|
|
|
region_id = TreeNodeMultipleChoiceFilter(
|
|
|
queryset=Region.objects.all(),
|
|
|
- field_name='terminations__site__region',
|
|
|
+ field_name='terminations___region',
|
|
|
lookup_expr='in',
|
|
|
label=_('Region (ID)'),
|
|
|
)
|
|
|
region = TreeNodeMultipleChoiceFilter(
|
|
|
queryset=Region.objects.all(),
|
|
|
- field_name='terminations__site__region',
|
|
|
+ field_name='terminations___region',
|
|
|
lookup_expr='in',
|
|
|
to_field_name='slug',
|
|
|
label=_('Region (slug)'),
|
|
|
)
|
|
|
site_group_id = TreeNodeMultipleChoiceFilter(
|
|
|
queryset=SiteGroup.objects.all(),
|
|
|
- field_name='terminations__site__group',
|
|
|
+ field_name='terminations___site_group',
|
|
|
lookup_expr='in',
|
|
|
label=_('Site group (ID)'),
|
|
|
)
|
|
|
site_group = TreeNodeMultipleChoiceFilter(
|
|
|
queryset=SiteGroup.objects.all(),
|
|
|
- field_name='terminations__site__group',
|
|
|
+ field_name='terminations___site_group',
|
|
|
lookup_expr='in',
|
|
|
to_field_name='slug',
|
|
|
label=_('Site group (slug)'),
|
|
|
)
|
|
|
site_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
- field_name='terminations__site',
|
|
|
+ field_name='terminations___site',
|
|
|
queryset=Site.objects.all(),
|
|
|
label=_('Site (ID)'),
|
|
|
)
|
|
|
site = django_filters.ModelMultipleChoiceFilter(
|
|
|
- field_name='terminations__site__slug',
|
|
|
+ field_name='terminations___site__slug',
|
|
|
queryset=Site.objects.all(),
|
|
|
to_field_name='slug',
|
|
|
label=_('Site (slug)'),
|
|
|
@@ -239,7 +245,9 @@ class CircuitFilterSet(NetBoxModelFilterSet, TenancyFilterSet, ContactModelFilte
|
|
|
|
|
|
class Meta:
|
|
|
model = Circuit
|
|
|
- fields = ('id', 'cid', 'description', 'install_date', 'termination_date', 'commit_rate')
|
|
|
+ fields = (
|
|
|
+ 'id', 'cid', 'description', 'install_date', 'termination_date', 'commit_rate', 'distance', 'distance_unit',
|
|
|
+ )
|
|
|
|
|
|
def search(self, queryset, name, value):
|
|
|
if not value.strip():
|
|
|
@@ -263,18 +271,60 @@ class CircuitTerminationFilterSet(NetBoxModelFilterSet, CabledObjectFilterSet):
|
|
|
queryset=Circuit.objects.all(),
|
|
|
label=_('Circuit'),
|
|
|
)
|
|
|
+ termination_type = ContentTypeFilter()
|
|
|
+ region_id = TreeNodeMultipleChoiceFilter(
|
|
|
+ queryset=Region.objects.all(),
|
|
|
+ field_name='_region',
|
|
|
+ lookup_expr='in',
|
|
|
+ label=_('Region (ID)'),
|
|
|
+ )
|
|
|
+ region = TreeNodeMultipleChoiceFilter(
|
|
|
+ queryset=Region.objects.all(),
|
|
|
+ field_name='_region',
|
|
|
+ lookup_expr='in',
|
|
|
+ to_field_name='slug',
|
|
|
+ label=_('Region (slug)'),
|
|
|
+ )
|
|
|
+ site_group_id = TreeNodeMultipleChoiceFilter(
|
|
|
+ queryset=SiteGroup.objects.all(),
|
|
|
+ field_name='_site_group',
|
|
|
+ lookup_expr='in',
|
|
|
+ label=_('Site group (ID)'),
|
|
|
+ )
|
|
|
+ site_group = TreeNodeMultipleChoiceFilter(
|
|
|
+ queryset=SiteGroup.objects.all(),
|
|
|
+ field_name='_site_group',
|
|
|
+ lookup_expr='in',
|
|
|
+ to_field_name='slug',
|
|
|
+ label=_('Site group (slug)'),
|
|
|
+ )
|
|
|
site_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
queryset=Site.objects.all(),
|
|
|
+ field_name='_site',
|
|
|
label=_('Site (ID)'),
|
|
|
)
|
|
|
site = django_filters.ModelMultipleChoiceFilter(
|
|
|
- field_name='site__slug',
|
|
|
+ field_name='_site__slug',
|
|
|
queryset=Site.objects.all(),
|
|
|
to_field_name='slug',
|
|
|
label=_('Site (slug)'),
|
|
|
)
|
|
|
+ location_id = TreeNodeMultipleChoiceFilter(
|
|
|
+ queryset=Location.objects.all(),
|
|
|
+ field_name='_location',
|
|
|
+ lookup_expr='in',
|
|
|
+ label=_('Location (ID)'),
|
|
|
+ )
|
|
|
+ location = TreeNodeMultipleChoiceFilter(
|
|
|
+ queryset=Location.objects.all(),
|
|
|
+ field_name='_location',
|
|
|
+ lookup_expr='in',
|
|
|
+ to_field_name='slug',
|
|
|
+ label=_('Location (slug)'),
|
|
|
+ )
|
|
|
provider_network_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
queryset=ProviderNetwork.objects.all(),
|
|
|
+ field_name='_provider_network',
|
|
|
label=_('ProviderNetwork (ID)'),
|
|
|
)
|
|
|
provider_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
@@ -292,8 +342,8 @@ class CircuitTerminationFilterSet(NetBoxModelFilterSet, CabledObjectFilterSet):
|
|
|
class Meta:
|
|
|
model = CircuitTermination
|
|
|
fields = (
|
|
|
- 'id', 'term_side', 'port_speed', 'upstream_speed', 'xconnect_id', 'description', 'mark_connected',
|
|
|
- 'pp_info', 'cable_end',
|
|
|
+ 'id', 'termination_id', 'term_side', 'port_speed', 'upstream_speed', 'xconnect_id', 'description',
|
|
|
+ 'mark_connected', 'pp_info', 'cable_end',
|
|
|
)
|
|
|
|
|
|
def search(self, queryset, name, value):
|
|
|
@@ -319,26 +369,36 @@ class CircuitGroupAssignmentFilterSet(NetBoxModelFilterSet):
|
|
|
method='search',
|
|
|
label=_('Search'),
|
|
|
)
|
|
|
- provider_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
- field_name='circuit__provider',
|
|
|
- queryset=Provider.objects.all(),
|
|
|
- label=_('Provider (ID)'),
|
|
|
- )
|
|
|
- provider = django_filters.ModelMultipleChoiceFilter(
|
|
|
- field_name='circuit__provider__slug',
|
|
|
- queryset=Provider.objects.all(),
|
|
|
- to_field_name='slug',
|
|
|
- label=_('Provider (slug)'),
|
|
|
+ member_type = ContentTypeFilter()
|
|
|
+ circuit = MultiValueCharFilter(
|
|
|
+ method='filter_circuit',
|
|
|
+ field_name='cid',
|
|
|
+ label=_('Circuit (CID)'),
|
|
|
)
|
|
|
- circuit_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
- queryset=Circuit.objects.all(),
|
|
|
+ circuit_id = MultiValueNumberFilter(
|
|
|
+ method='filter_circuit',
|
|
|
+ field_name='pk',
|
|
|
label=_('Circuit (ID)'),
|
|
|
)
|
|
|
- circuit = django_filters.ModelMultipleChoiceFilter(
|
|
|
- field_name='circuit__cid',
|
|
|
- queryset=Circuit.objects.all(),
|
|
|
- to_field_name='cid',
|
|
|
- label=_('Circuit (CID)'),
|
|
|
+ virtual_circuit = MultiValueCharFilter(
|
|
|
+ method='filter_virtual_circuit',
|
|
|
+ field_name='cid',
|
|
|
+ label=_('Virtual circuit (CID)'),
|
|
|
+ )
|
|
|
+ virtual_circuit_id = MultiValueNumberFilter(
|
|
|
+ method='filter_virtual_circuit',
|
|
|
+ field_name='pk',
|
|
|
+ label=_('Virtual circuit (ID)'),
|
|
|
+ )
|
|
|
+ provider = MultiValueCharFilter(
|
|
|
+ method='filter_provider',
|
|
|
+ field_name='slug',
|
|
|
+ label=_('Provider (name)'),
|
|
|
+ )
|
|
|
+ provider_id = MultiValueNumberFilter(
|
|
|
+ method='filter_provider',
|
|
|
+ field_name='pk',
|
|
|
+ label=_('Provider (ID)'),
|
|
|
)
|
|
|
group_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
queryset=CircuitGroup.objects.all(),
|
|
|
@@ -353,12 +413,173 @@ class CircuitGroupAssignmentFilterSet(NetBoxModelFilterSet):
|
|
|
|
|
|
class Meta:
|
|
|
model = CircuitGroupAssignment
|
|
|
- fields = ('id', 'priority')
|
|
|
+ fields = ('id', 'member_id', 'priority')
|
|
|
|
|
|
def search(self, queryset, name, value):
|
|
|
if not value.strip():
|
|
|
return queryset
|
|
|
return queryset.filter(
|
|
|
- Q(circuit__cid__icontains=value) |
|
|
|
+ Q(member__cid__icontains=value) |
|
|
|
Q(group__name__icontains=value)
|
|
|
)
|
|
|
+
|
|
|
+ def filter_circuit(self, queryset, name, value):
|
|
|
+ circuits = Circuit.objects.filter(**{f'{name}__in': value})
|
|
|
+ if not circuits.exists():
|
|
|
+ return queryset.none()
|
|
|
+ return queryset.filter(
|
|
|
+ Q(
|
|
|
+ member_type=ContentType.objects.get_for_model(Circuit),
|
|
|
+ member_id__in=circuits
|
|
|
+ )
|
|
|
+ )
|
|
|
+
|
|
|
+ def filter_virtual_circuit(self, queryset, name, value):
|
|
|
+ virtual_circuits = VirtualCircuit.objects.filter(**{f'{name}__in': value})
|
|
|
+ if not virtual_circuits.exists():
|
|
|
+ return queryset.none()
|
|
|
+ return queryset.filter(
|
|
|
+ Q(
|
|
|
+ member_type=ContentType.objects.get_for_model(VirtualCircuit),
|
|
|
+ member_id__in=virtual_circuits
|
|
|
+ )
|
|
|
+ )
|
|
|
+
|
|
|
+ def filter_provider(self, queryset, name, value):
|
|
|
+ providers = Provider.objects.filter(**{f'{name}__in': value})
|
|
|
+ if not providers.exists():
|
|
|
+ return queryset.none()
|
|
|
+ circuits = Circuit.objects.filter(provider__in=providers)
|
|
|
+ virtual_circuits = VirtualCircuit.objects.filter(provider_network__provider__in=providers)
|
|
|
+ return queryset.filter(
|
|
|
+ Q(
|
|
|
+ member_type=ContentType.objects.get_for_model(Circuit),
|
|
|
+ member_id__in=circuits
|
|
|
+ ) |
|
|
|
+ Q(
|
|
|
+ member_type=ContentType.objects.get_for_model(VirtualCircuit),
|
|
|
+ member_id__in=virtual_circuits
|
|
|
+ )
|
|
|
+ )
|
|
|
+
|
|
|
+
|
|
|
+class VirtualCircuitTypeFilterSet(OrganizationalModelFilterSet):
|
|
|
+
|
|
|
+ class Meta:
|
|
|
+ model = VirtualCircuitType
|
|
|
+ fields = ('id', 'name', 'slug', 'color', 'description')
|
|
|
+
|
|
|
+
|
|
|
+class VirtualCircuitFilterSet(NetBoxModelFilterSet, TenancyFilterSet):
|
|
|
+ provider_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ field_name='provider_network__provider',
|
|
|
+ queryset=Provider.objects.all(),
|
|
|
+ label=_('Provider (ID)'),
|
|
|
+ )
|
|
|
+ provider = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ field_name='provider_network__provider__slug',
|
|
|
+ queryset=Provider.objects.all(),
|
|
|
+ to_field_name='slug',
|
|
|
+ label=_('Provider (slug)'),
|
|
|
+ )
|
|
|
+ provider_account_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ field_name='provider_account',
|
|
|
+ queryset=ProviderAccount.objects.all(),
|
|
|
+ label=_('Provider account (ID)'),
|
|
|
+ )
|
|
|
+ provider_account = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ field_name='provider_account__account',
|
|
|
+ queryset=Provider.objects.all(),
|
|
|
+ to_field_name='account',
|
|
|
+ label=_('Provider account (account)'),
|
|
|
+ )
|
|
|
+ provider_network_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ queryset=ProviderNetwork.objects.all(),
|
|
|
+ label=_('Provider network (ID)'),
|
|
|
+ )
|
|
|
+ type_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ queryset=VirtualCircuitType.objects.all(),
|
|
|
+ label=_('Virtual circuit type (ID)'),
|
|
|
+ )
|
|
|
+ type = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ field_name='type__slug',
|
|
|
+ queryset=VirtualCircuitType.objects.all(),
|
|
|
+ to_field_name='slug',
|
|
|
+ label=_('Virtual circuit type (slug)'),
|
|
|
+ )
|
|
|
+ status = django_filters.MultipleChoiceFilter(
|
|
|
+ choices=CircuitStatusChoices,
|
|
|
+ null_value=None
|
|
|
+ )
|
|
|
+
|
|
|
+ class Meta:
|
|
|
+ model = VirtualCircuit
|
|
|
+ fields = ('id', 'cid', 'description')
|
|
|
+
|
|
|
+ def search(self, queryset, name, value):
|
|
|
+ if not value.strip():
|
|
|
+ return queryset
|
|
|
+ return queryset.filter(
|
|
|
+ Q(cid__icontains=value) |
|
|
|
+ Q(description__icontains=value) |
|
|
|
+ Q(comments__icontains=value)
|
|
|
+ ).distinct()
|
|
|
+
|
|
|
+
|
|
|
+class VirtualCircuitTerminationFilterSet(NetBoxModelFilterSet):
|
|
|
+ q = django_filters.CharFilter(
|
|
|
+ method='search',
|
|
|
+ label=_('Search'),
|
|
|
+ )
|
|
|
+ virtual_circuit_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ queryset=VirtualCircuit.objects.all(),
|
|
|
+ label=_('Virtual circuit'),
|
|
|
+ )
|
|
|
+ role = django_filters.MultipleChoiceFilter(
|
|
|
+ choices=VirtualCircuitTerminationRoleChoices,
|
|
|
+ null_value=None
|
|
|
+ )
|
|
|
+ provider_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ field_name='virtual_circuit__provider_network__provider',
|
|
|
+ queryset=Provider.objects.all(),
|
|
|
+ label=_('Provider (ID)'),
|
|
|
+ )
|
|
|
+ provider = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ field_name='virtual_circuit__provider_network__provider__slug',
|
|
|
+ queryset=Provider.objects.all(),
|
|
|
+ to_field_name='slug',
|
|
|
+ label=_('Provider (slug)'),
|
|
|
+ )
|
|
|
+ provider_account_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ field_name='virtual_circuit__provider_account',
|
|
|
+ queryset=ProviderAccount.objects.all(),
|
|
|
+ label=_('Provider account (ID)'),
|
|
|
+ )
|
|
|
+ provider_account = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ field_name='virtual_circuit__provider_account__account',
|
|
|
+ queryset=ProviderAccount.objects.all(),
|
|
|
+ to_field_name='account',
|
|
|
+ label=_('Provider account (account)'),
|
|
|
+ )
|
|
|
+ provider_network_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ queryset=ProviderNetwork.objects.all(),
|
|
|
+ field_name='virtual_circuit__provider_network',
|
|
|
+ label=_('Provider network (ID)'),
|
|
|
+ )
|
|
|
+ interface_id = django_filters.ModelMultipleChoiceFilter(
|
|
|
+ queryset=Interface.objects.all(),
|
|
|
+ field_name='interface',
|
|
|
+ label=_('Interface (ID)'),
|
|
|
+ )
|
|
|
+
|
|
|
+ class Meta:
|
|
|
+ model = VirtualCircuitTermination
|
|
|
+ fields = ('id', 'interface_id', 'description')
|
|
|
+
|
|
|
+ def search(self, queryset, name, value):
|
|
|
+ if not value.strip():
|
|
|
+ return queryset
|
|
|
+ return queryset.filter(
|
|
|
+ Q(virtual_circuit__cid__icontains=value) |
|
|
|
+ Q(description__icontains=value)
|
|
|
+ ).distinct()
|