mirror of
https://github.com/netbox-community/netbox.git
synced 2024-05-10 07:54:54 +00:00
@ -265,10 +265,11 @@ class DeviceListView(CustomFieldModelAPIView, generics.ListAPIView):
|
|||||||
"""
|
"""
|
||||||
List devices (filterable)
|
List devices (filterable)
|
||||||
"""
|
"""
|
||||||
queryset = Device.objects.select_related('device_type__manufacturer', 'device_role', 'tenant', 'platform',
|
queryset = Device.objects.select_related(
|
||||||
'rack__site', 'parent_bay').prefetch_related('primary_ip4__nat_outside',
|
'device_type__manufacturer', 'device_role', 'tenant', 'platform', 'site', 'rack', 'parent_bay'
|
||||||
'primary_ip6__nat_outside',
|
).prefetch_related(
|
||||||
'custom_field_values__field')
|
'primary_ip4__nat_outside', 'primary_ip6__nat_outside', 'custom_field_values__field'
|
||||||
|
)
|
||||||
serializer_class = serializers.DeviceSerializer
|
serializer_class = serializers.DeviceSerializer
|
||||||
filter_class = filters.DeviceFilter
|
filter_class = filters.DeviceFilter
|
||||||
renderer_classes = api_settings.DEFAULT_RENDERER_CLASSES + [BINDZoneRenderer, FlatJSONRenderer]
|
renderer_classes = api_settings.DEFAULT_RENDERER_CLASSES + [BINDZoneRenderer, FlatJSONRenderer]
|
||||||
@ -278,8 +279,9 @@ class DeviceDetailView(CustomFieldModelAPIView, generics.RetrieveAPIView):
|
|||||||
"""
|
"""
|
||||||
Retrieve a single device
|
Retrieve a single device
|
||||||
"""
|
"""
|
||||||
queryset = Device.objects.select_related('device_type__manufacturer', 'device_role', 'tenant', 'platform',
|
queryset = Device.objects.select_related(
|
||||||
'rack__site', 'parent_bay').prefetch_related('custom_field_values__field')
|
'device_type__manufacturer', 'device_role', 'tenant', 'platform', 'site', 'rack', 'parent_bay'
|
||||||
|
).prefetch_related('custom_field_values__field')
|
||||||
serializer_class = serializers.DeviceSerializer
|
serializer_class = serializers.DeviceSerializer
|
||||||
|
|
||||||
|
|
||||||
|
@ -934,28 +934,29 @@ class ConsolePortConnectionForm(BootstrapMixin, forms.ModelForm):
|
|||||||
if not self.instance.pk:
|
if not self.instance.pk:
|
||||||
raise RuntimeError("ConsolePortConnectionForm must be initialized with an existing ConsolePort instance.")
|
raise RuntimeError("ConsolePortConnectionForm must be initialized with an existing ConsolePort instance.")
|
||||||
|
|
||||||
self.initial['site'] = self.instance.device.site
|
# Initialize rack choices if site is set
|
||||||
self.fields['rack'].queryset = Rack.objects.filter(site=self.instance.device.site)
|
if self.initial.get('site'):
|
||||||
self.fields['cs_port'].required = True
|
self.fields['rack'].queryset = Rack.objects.filter(site=self.initial['site'])
|
||||||
self.fields['connection_status'].choices = CONNECTION_STATUS_CHOICES
|
|
||||||
|
|
||||||
# Initialize console server choices
|
|
||||||
if self.is_bound and self.data.get('rack'):
|
|
||||||
self.fields['console_server'].queryset = Device.objects.filter(rack=self.data['rack'],
|
|
||||||
device_type__is_console_server=True)
|
|
||||||
elif self.initial.get('rack'):
|
|
||||||
self.fields['console_server'].queryset = Device.objects.filter(rack=self.initial['rack'],
|
|
||||||
device_type__is_console_server=True)
|
|
||||||
else:
|
else:
|
||||||
self.fields['console_server'].queryset = Device.objects.filter(site=self.instance.device.site,
|
self.fields['rack'].choices = []
|
||||||
rack__isnull=True,
|
|
||||||
device_type__is_console_server=True)
|
|
||||||
|
|
||||||
# Initialize CS port choices
|
# Initialize console_server choices if rack or site is set
|
||||||
if self.is_bound:
|
if self.initial.get('rack'):
|
||||||
self.fields['cs_port'].queryset = ConsoleServerPort.objects.filter(device__pk=self.data['console_server'])
|
self.fields['console_server'].queryset = Device.objects.filter(
|
||||||
elif self.initial.get('console_server', None):
|
rack=self.initial['rack'], device_type__is_console_server=True
|
||||||
self.fields['cs_port'].queryset = ConsoleServerPort.objects.filter(device__pk=self.initial['console_server'])
|
)
|
||||||
|
elif self.initial.get('site'):
|
||||||
|
self.fields['console_server'].queryset = Device.objects.filter(
|
||||||
|
site=self.initial['site'], rack__isnull=True, device_type__is_console_server=True
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
self.fields['console_server'].choices = []
|
||||||
|
|
||||||
|
# Initialize CS port choices if console_server is set
|
||||||
|
if self.initial.get('console_server'):
|
||||||
|
self.fields['cs_port'].queryset = ConsoleServerPort.objects.filter(
|
||||||
|
device=self.initial['console_server']
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
self.fields['cs_port'].choices = []
|
self.fields['cs_port'].choices = []
|
||||||
|
|
||||||
@ -1033,27 +1034,27 @@ class ConsoleServerPortConnectionForm(BootstrapMixin, forms.Form):
|
|||||||
'connection_status': 'Status',
|
'connection_status': 'Status',
|
||||||
}
|
}
|
||||||
|
|
||||||
def __init__(self, consoleserverport, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
|
|
||||||
super(ConsoleServerPortConnectionForm, self).__init__(*args, **kwargs)
|
super(ConsoleServerPortConnectionForm, self).__init__(*args, **kwargs)
|
||||||
|
|
||||||
self.initial['site'] = consoleserverport.device.site
|
# Initialize rack choices if site is set
|
||||||
self.fields['rack'].queryset = Rack.objects.filter(site=consoleserverport.device.site)
|
if self.initial.get('site'):
|
||||||
|
self.fields['rack'].queryset = Rack.objects.filter(site=self.initial['site'])
|
||||||
# Initialize device choices
|
|
||||||
if self.is_bound and self.data.get('rack'):
|
|
||||||
self.fields['device'].queryset = Device.objects.filter(rack=self.data['rack'])
|
|
||||||
elif self.initial.get('rack', None):
|
|
||||||
self.fields['device'].queryset = Device.objects.filter(rack=self.initial['rack'])
|
|
||||||
else:
|
else:
|
||||||
self.fields['device'].queryset = Device.objects.filter(site=consoleserverport.device.site,
|
self.fields['rack'].choices = []
|
||||||
rack__isnull=True)
|
|
||||||
|
|
||||||
# Initialize port choices
|
# Initialize device choices if rack or site is set
|
||||||
if self.is_bound:
|
if self.initial.get('rack'):
|
||||||
self.fields['port'].queryset = ConsolePort.objects.filter(device__pk=self.data['device'])
|
self.fields['device'].queryset = Device.objects.filter(rack=self.initial['rack'])
|
||||||
elif self.initial.get('device', None):
|
elif self.initial.get('site'):
|
||||||
self.fields['port'].queryset = ConsolePort.objects.filter(device_pk=self.initial['device'])
|
self.fields['device'].queryset = Device.objects.filter(site=self.initial['site'], rack__isnull=True)
|
||||||
|
else:
|
||||||
|
self.fields['device'].choices = []
|
||||||
|
|
||||||
|
# Initialize port choices if device is set
|
||||||
|
if self.initial.get('device'):
|
||||||
|
self.fields['port'].queryset = ConsolePort.objects.filter(device=self.initial['device'])
|
||||||
else:
|
else:
|
||||||
self.fields['port'].choices = []
|
self.fields['port'].choices = []
|
||||||
|
|
||||||
@ -1201,28 +1202,27 @@ class PowerPortConnectionForm(BootstrapMixin, forms.ModelForm):
|
|||||||
if not self.instance.pk:
|
if not self.instance.pk:
|
||||||
raise RuntimeError("PowerPortConnectionForm must be initialized with an existing PowerPort instance.")
|
raise RuntimeError("PowerPortConnectionForm must be initialized with an existing PowerPort instance.")
|
||||||
|
|
||||||
self.initial['site'] = self.instance.device.site
|
# Initialize rack choices if site is set
|
||||||
self.fields['rack'].queryset = Rack.objects.filter(site=self.instance.device.site)
|
if self.initial.get('site'):
|
||||||
self.fields['power_outlet'].required = True
|
self.fields['rack'].queryset = Rack.objects.filter(site=self.initial['site'])
|
||||||
self.fields['connection_status'].choices = CONNECTION_STATUS_CHOICES
|
|
||||||
|
|
||||||
# Initialize PDU choices
|
|
||||||
if self.is_bound and self.data.get('rack'):
|
|
||||||
self.fields['pdu'].queryset = Device.objects.filter(rack=self.data['rack'],
|
|
||||||
device_type__is_pdu=True)
|
|
||||||
elif self.initial.get('rack', None):
|
|
||||||
self.fields['pdu'].queryset = Device.objects.filter(rack=self.initial['rack'],
|
|
||||||
device_type__is_pdu=True)
|
|
||||||
else:
|
else:
|
||||||
self.fields['pdu'].queryset = Device.objects.filter(site=self.instance.device.site,
|
self.fields['rack'].choices = []
|
||||||
rack__isnull=True,
|
|
||||||
device_type__is_pdu=True)
|
|
||||||
|
|
||||||
# Initialize power outlet choices
|
# Initialize pdu choices if rack or site is set
|
||||||
if self.is_bound:
|
if self.initial.get('rack'):
|
||||||
self.fields['power_outlet'].queryset = PowerOutlet.objects.filter(device__pk=self.data['pdu'])
|
self.fields['pdu'].queryset = Device.objects.filter(
|
||||||
elif self.initial.get('pdu', None):
|
rack=self.initial['rack'], device_type__is_pdu=True
|
||||||
self.fields['power_outlet'].queryset = PowerOutlet.objects.filter(device__pk=self.initial['pdu'])
|
)
|
||||||
|
elif self.initial.get('site'):
|
||||||
|
self.fields['pdu'].queryset = Device.objects.filter(
|
||||||
|
site=self.initial['site'], rack__isnull=True, device_type__is_pdu=True
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
self.fields['pdu'].choices = []
|
||||||
|
|
||||||
|
# Initialize power outlet choices if pdu is set
|
||||||
|
if self.initial.get('pdu'):
|
||||||
|
self.fields['power_outlet'].queryset = PowerOutlet.objects.filter(device=self.initial['pdu'])
|
||||||
else:
|
else:
|
||||||
self.fields['power_outlet'].choices = []
|
self.fields['power_outlet'].choices = []
|
||||||
|
|
||||||
@ -1300,27 +1300,27 @@ class PowerOutletConnectionForm(BootstrapMixin, forms.Form):
|
|||||||
'connection_status': 'Status',
|
'connection_status': 'Status',
|
||||||
}
|
}
|
||||||
|
|
||||||
def __init__(self, poweroutlet, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
|
|
||||||
super(PowerOutletConnectionForm, self).__init__(*args, **kwargs)
|
super(PowerOutletConnectionForm, self).__init__(*args, **kwargs)
|
||||||
|
|
||||||
self.initial['site'] = poweroutlet.device.site
|
# Initialize rack choices if site is set
|
||||||
self.fields['rack'].queryset = Rack.objects.filter(site=poweroutlet.device.site)
|
if self.initial.get('site'):
|
||||||
|
self.fields['rack'].queryset = Rack.objects.filter(site=self.initial['site'])
|
||||||
# Initialize device choices
|
|
||||||
if self.is_bound and self.data.get('rack'):
|
|
||||||
self.fields['device'].queryset = Device.objects.filter(rack=self.data['rack'])
|
|
||||||
elif self.initial.get('rack', None):
|
|
||||||
self.fields['device'].queryset = Device.objects.filter(rack=self.initial['rack'])
|
|
||||||
else:
|
else:
|
||||||
self.fields['device'].queryset = Device.objects.filter(site=poweroutlet.device.site,
|
self.fields['rack'].choices = []
|
||||||
rack__isnull=True)
|
|
||||||
|
|
||||||
# Initialize port choices
|
# Initialize device choices if rack or site is set
|
||||||
if self.is_bound:
|
if self.initial.get('rack'):
|
||||||
self.fields['port'].queryset = PowerPort.objects.filter(device__pk=self.data['device'])
|
self.fields['device'].queryset = Device.objects.filter(rack=self.initial['rack'])
|
||||||
elif self.initial.get('device', None):
|
elif self.initial.get('site'):
|
||||||
self.fields['port'].queryset = PowerPort.objects.filter(device_pk=self.initial['device'])
|
self.fields['device'].queryset = Device.objects.filter(site=self.initial['site'], rack__isnull=True)
|
||||||
|
else:
|
||||||
|
self.fields['device'].choices = []
|
||||||
|
|
||||||
|
# Initialize port choices if device is set
|
||||||
|
if self.initial.get('device'):
|
||||||
|
self.fields['port'].queryset = PowerPort.objects.filter(device=self.initial['device'])
|
||||||
else:
|
else:
|
||||||
self.fields['port'].choices = []
|
self.fields['port'].choices = []
|
||||||
|
|
||||||
|
@ -293,7 +293,7 @@ class Site(CreatedUpdatedModel, CustomFieldModel):
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def count_devices(self):
|
def count_devices(self):
|
||||||
return Device.objects.filter(rack__site=self).count()
|
return Device.objects.filter(site=self).count()
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def count_circuits(self):
|
def count_circuits(self):
|
||||||
|
@ -12,11 +12,11 @@ from .models import (
|
|||||||
|
|
||||||
REGION_LINK = """
|
REGION_LINK = """
|
||||||
{% if record.get_children %}
|
{% if record.get_children %}
|
||||||
<span style="padding-left: {{ record.get_ancestors|length }}0px "><i class="fa fa-caret-right"></i></a>
|
<span style="padding-left: {{ record.get_ancestors|length }}0px "><i class="fa fa-caret-right"></i>
|
||||||
{% else %}
|
{% else %}
|
||||||
<span style="padding-left: {{ record.get_ancestors|length }}9px">
|
<span style="padding-left: {{ record.get_ancestors|length }}9px">
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{{ record.name }}
|
<a href="{% url 'dcim:site_list' %}?region={{ record.slug }}">{{ record.name }}</a>
|
||||||
</span>
|
</span>
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
@ -171,7 +171,7 @@ def site(request, slug):
|
|||||||
site = get_object_or_404(Site.objects.select_related('region', 'tenant__group'), slug=slug)
|
site = get_object_or_404(Site.objects.select_related('region', 'tenant__group'), slug=slug)
|
||||||
stats = {
|
stats = {
|
||||||
'rack_count': Rack.objects.filter(site=site).count(),
|
'rack_count': Rack.objects.filter(site=site).count(),
|
||||||
'device_count': Device.objects.filter(rack__site=site).count(),
|
'device_count': Device.objects.filter(site=site).count(),
|
||||||
'prefix_count': Prefix.objects.filter(site=site).count(),
|
'prefix_count': Prefix.objects.filter(site=site).count(),
|
||||||
'vlan_count': VLAN.objects.filter(site=site).count(),
|
'vlan_count': VLAN.objects.filter(site=site).count(),
|
||||||
'circuit_count': Circuit.objects.filter(terminations__site=site).count(),
|
'circuit_count': Circuit.objects.filter(terminations__site=site).count(),
|
||||||
@ -844,7 +844,9 @@ def consoleport_connect(request, pk):
|
|||||||
|
|
||||||
else:
|
else:
|
||||||
form = forms.ConsolePortConnectionForm(instance=consoleport, initial={
|
form = forms.ConsolePortConnectionForm(instance=consoleport, initial={
|
||||||
'rack': consoleport.device.rack,
|
'site': request.GET.get('site', consoleport.device.site),
|
||||||
|
'rack': request.GET.get('rack', None),
|
||||||
|
'console_server': request.GET.get('console_server', None),
|
||||||
'connection_status': CONNECTION_STATUS_CONNECTED,
|
'connection_status': CONNECTION_STATUS_CONNECTED,
|
||||||
})
|
})
|
||||||
|
|
||||||
@ -927,7 +929,7 @@ def consoleserverport_connect(request, pk):
|
|||||||
consoleserverport = get_object_or_404(ConsoleServerPort, pk=pk)
|
consoleserverport = get_object_or_404(ConsoleServerPort, pk=pk)
|
||||||
|
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
form = forms.ConsoleServerPortConnectionForm(consoleserverport, request.POST)
|
form = forms.ConsoleServerPortConnectionForm(request.POST)
|
||||||
if form.is_valid():
|
if form.is_valid():
|
||||||
consoleport = form.cleaned_data['port']
|
consoleport = form.cleaned_data['port']
|
||||||
consoleport.cs_port = consoleserverport
|
consoleport.cs_port = consoleserverport
|
||||||
@ -942,7 +944,12 @@ def consoleserverport_connect(request, pk):
|
|||||||
return redirect('dcim:device', pk=consoleserverport.device.pk)
|
return redirect('dcim:device', pk=consoleserverport.device.pk)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
form = forms.ConsoleServerPortConnectionForm(consoleserverport, initial={'rack': consoleserverport.device.rack})
|
form = forms.ConsoleServerPortConnectionForm(initial={
|
||||||
|
'site': request.GET.get('site', consoleserverport.device.site),
|
||||||
|
'rack': request.GET.get('rack', None),
|
||||||
|
'device': request.GET.get('device', None),
|
||||||
|
'connection_status': CONNECTION_STATUS_CONNECTED,
|
||||||
|
})
|
||||||
|
|
||||||
return render(request, 'dcim/consoleserverport_connect.html', {
|
return render(request, 'dcim/consoleserverport_connect.html', {
|
||||||
'consoleserverport': consoleserverport,
|
'consoleserverport': consoleserverport,
|
||||||
@ -1030,7 +1037,9 @@ def powerport_connect(request, pk):
|
|||||||
|
|
||||||
else:
|
else:
|
||||||
form = forms.PowerPortConnectionForm(instance=powerport, initial={
|
form = forms.PowerPortConnectionForm(instance=powerport, initial={
|
||||||
'rack': powerport.device.rack,
|
'site': request.GET.get('site', powerport.device.site),
|
||||||
|
'rack': request.GET.get('rack', None),
|
||||||
|
'pdu': request.GET.get('pdu', None),
|
||||||
'connection_status': CONNECTION_STATUS_CONNECTED,
|
'connection_status': CONNECTION_STATUS_CONNECTED,
|
||||||
})
|
})
|
||||||
|
|
||||||
@ -1113,7 +1122,7 @@ def poweroutlet_connect(request, pk):
|
|||||||
poweroutlet = get_object_or_404(PowerOutlet, pk=pk)
|
poweroutlet = get_object_or_404(PowerOutlet, pk=pk)
|
||||||
|
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
form = forms.PowerOutletConnectionForm(poweroutlet, request.POST)
|
form = forms.PowerOutletConnectionForm(request.POST)
|
||||||
if form.is_valid():
|
if form.is_valid():
|
||||||
powerport = form.cleaned_data['port']
|
powerport = form.cleaned_data['port']
|
||||||
powerport.power_outlet = poweroutlet
|
powerport.power_outlet = poweroutlet
|
||||||
@ -1128,7 +1137,12 @@ def poweroutlet_connect(request, pk):
|
|||||||
return redirect('dcim:device', pk=poweroutlet.device.pk)
|
return redirect('dcim:device', pk=poweroutlet.device.pk)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
form = forms.PowerOutletConnectionForm(poweroutlet, initial={'rack': poweroutlet.device.rack})
|
form = forms.PowerOutletConnectionForm(initial={
|
||||||
|
'site': request.GET.get('site', poweroutlet.device.site),
|
||||||
|
'rack': request.GET.get('rack', None),
|
||||||
|
'device': request.GET.get('device', None),
|
||||||
|
'connection_status': CONNECTION_STATUS_CONNECTED,
|
||||||
|
})
|
||||||
|
|
||||||
return render(request, 'dcim/poweroutlet_connect.html', {
|
return render(request, 'dcim/poweroutlet_connect.html', {
|
||||||
'poweroutlet': poweroutlet,
|
'poweroutlet': poweroutlet,
|
||||||
|
@ -49,7 +49,7 @@ class Command(BaseCommand):
|
|||||||
self.stdout.write("Running inventory for these sites: {}".format(', '.join(site_names)))
|
self.stdout.write("Running inventory for these sites: {}".format(', '.join(site_names)))
|
||||||
else:
|
else:
|
||||||
raise CommandError("One or more sites specified but none found.")
|
raise CommandError("One or more sites specified but none found.")
|
||||||
device_list = device_list.filter(rack__site__in=sites)
|
device_list = device_list.filter(site__in=sites)
|
||||||
|
|
||||||
# --name: Filter devices by name matching a regex
|
# --name: Filter devices by name matching a regex
|
||||||
if options['name']:
|
if options['name']:
|
||||||
|
@ -6,7 +6,7 @@ from django.db import models
|
|||||||
from .formfields import IPFormField
|
from .formfields import IPFormField
|
||||||
from .lookups import (
|
from .lookups import (
|
||||||
EndsWith, IEndsWith, IRegex, IStartsWith, NetContained, NetContainedOrEqual, NetContains, NetContainsOrEquals,
|
EndsWith, IEndsWith, IRegex, IStartsWith, NetContained, NetContainedOrEqual, NetContains, NetContainsOrEquals,
|
||||||
NetHost, NetMaskLength, Regex, StartsWith,
|
NetHost, NetHostContained, NetMaskLength, Regex, StartsWith,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@ -66,7 +66,6 @@ IPNetworkField.register_lookup(NetContained)
|
|||||||
IPNetworkField.register_lookup(NetContainedOrEqual)
|
IPNetworkField.register_lookup(NetContainedOrEqual)
|
||||||
IPNetworkField.register_lookup(NetContains)
|
IPNetworkField.register_lookup(NetContains)
|
||||||
IPNetworkField.register_lookup(NetContainsOrEquals)
|
IPNetworkField.register_lookup(NetContainsOrEquals)
|
||||||
IPNetworkField.register_lookup(NetHost)
|
|
||||||
IPNetworkField.register_lookup(NetMaskLength)
|
IPNetworkField.register_lookup(NetMaskLength)
|
||||||
|
|
||||||
|
|
||||||
@ -91,4 +90,5 @@ IPAddressField.register_lookup(NetContainedOrEqual)
|
|||||||
IPAddressField.register_lookup(NetContains)
|
IPAddressField.register_lookup(NetContains)
|
||||||
IPAddressField.register_lookup(NetContainsOrEquals)
|
IPAddressField.register_lookup(NetContainsOrEquals)
|
||||||
IPAddressField.register_lookup(NetHost)
|
IPAddressField.register_lookup(NetHost)
|
||||||
|
IPAddressField.register_lookup(NetHostContained)
|
||||||
IPAddressField.register_lookup(NetMaskLength)
|
IPAddressField.register_lookup(NetMaskLength)
|
||||||
|
@ -254,7 +254,7 @@ class IPAddressFilter(CustomFieldFilterSet, django_filters.FilterSet):
|
|||||||
return queryset
|
return queryset
|
||||||
try:
|
try:
|
||||||
query = str(IPNetwork(value.strip()).cidr)
|
query = str(IPNetwork(value.strip()).cidr)
|
||||||
return queryset.filter(address__net_contained_or_equal=query)
|
return queryset.filter(address__net_host_contained=query)
|
||||||
except AddrFormatError:
|
except AddrFormatError:
|
||||||
return queryset.none()
|
return queryset.none()
|
||||||
|
|
||||||
|
@ -333,9 +333,11 @@ class IPAddressForm(BootstrapMixin, CustomFieldForm):
|
|||||||
self.initial['nat_site'] = self.instance.nat_inside.interface.device.site.pk
|
self.initial['nat_site'] = self.instance.nat_inside.interface.device.site.pk
|
||||||
self.initial['nat_device'] = self.instance.nat_inside.interface.device.pk
|
self.initial['nat_device'] = self.instance.nat_inside.interface.device.pk
|
||||||
self.fields['nat_device'].queryset = Device.objects.filter(
|
self.fields['nat_device'].queryset = Device.objects.filter(
|
||||||
rack__site=nat_inside.interface.device.site)
|
site=nat_inside.interface.device.site
|
||||||
|
)
|
||||||
self.fields['nat_inside'].queryset = IPAddress.objects.filter(
|
self.fields['nat_inside'].queryset = IPAddress.objects.filter(
|
||||||
interface__device=nat_inside.interface.device)
|
interface__device=nat_inside.interface.device
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
self.fields['nat_inside'].queryset = IPAddress.objects.filter(pk=nat_inside.pk)
|
self.fields['nat_inside'].queryset = IPAddress.objects.filter(pk=nat_inside.pk)
|
||||||
|
|
||||||
@ -343,9 +345,9 @@ class IPAddressForm(BootstrapMixin, CustomFieldForm):
|
|||||||
|
|
||||||
# Initialize nat_device choices if nat_site is set
|
# Initialize nat_device choices if nat_site is set
|
||||||
if self.is_bound and self.data.get('nat_site'):
|
if self.is_bound and self.data.get('nat_site'):
|
||||||
self.fields['nat_device'].queryset = Device.objects.filter(rack__site__pk=self.data['nat_site'])
|
self.fields['nat_device'].queryset = Device.objects.filter(site__pk=self.data['nat_site'])
|
||||||
elif self.initial.get('nat_site'):
|
elif self.initial.get('nat_site'):
|
||||||
self.fields['nat_device'].queryset = Device.objects.filter(rack__site=self.initial['nat_site'])
|
self.fields['nat_device'].queryset = Device.objects.filter(site=self.initial['nat_site'])
|
||||||
else:
|
else:
|
||||||
self.fields['nat_device'].choices = []
|
self.fields['nat_device'].choices = []
|
||||||
|
|
||||||
|
@ -89,6 +89,20 @@ class NetHost(Lookup):
|
|||||||
return 'HOST(%s) = %s' % (lhs, rhs), params
|
return 'HOST(%s) = %s' % (lhs, rhs), params
|
||||||
|
|
||||||
|
|
||||||
|
class NetHostContained(Lookup):
|
||||||
|
"""
|
||||||
|
Check for the host portion of an IP address without regard to its mask. This allows us to find e.g. 192.0.2.1/24
|
||||||
|
when specifying a parent prefix of 192.0.2.0/26.
|
||||||
|
"""
|
||||||
|
lookup_name = 'net_host_contained'
|
||||||
|
|
||||||
|
def as_sql(self, qn, connection):
|
||||||
|
lhs, lhs_params = self.process_lhs(qn, connection)
|
||||||
|
rhs, rhs_params = self.process_rhs(qn, connection)
|
||||||
|
params = lhs_params + rhs_params
|
||||||
|
return 'CAST(HOST(%s) AS INET) << %s' % (lhs, rhs), params
|
||||||
|
|
||||||
|
|
||||||
class NetMaskLength(Transform):
|
class NetMaskLength(Transform):
|
||||||
lookup_name = 'net_mask_length'
|
lookup_name = 'net_mask_length'
|
||||||
function = 'MASKLEN'
|
function = 'MASKLEN'
|
||||||
|
@ -267,7 +267,7 @@ class PrefixQuerySet(NullsFirstQuerySet):
|
|||||||
p.depth = len(stack) - 1
|
p.depth = len(stack) - 1
|
||||||
if limit is None:
|
if limit is None:
|
||||||
return queryset
|
return queryset
|
||||||
return filter(lambda p: p.depth <= limit, queryset)
|
return list(filter(lambda p: p.depth <= limit, queryset))
|
||||||
|
|
||||||
|
|
||||||
@python_2_unicode_compatible
|
@python_2_unicode_compatible
|
||||||
|
@ -403,7 +403,7 @@ def prefix(request, pk):
|
|||||||
aggregate = None
|
aggregate = None
|
||||||
|
|
||||||
# Count child IP addresses
|
# Count child IP addresses
|
||||||
ipaddress_count = IPAddress.objects.filter(vrf=prefix.vrf, address__net_contained_or_equal=str(prefix.prefix))\
|
ipaddress_count = IPAddress.objects.filter(vrf=prefix.vrf, address__net_host_contained=str(prefix.prefix))\
|
||||||
.count()
|
.count()
|
||||||
|
|
||||||
# Parent prefixes table
|
# Parent prefixes table
|
||||||
@ -420,13 +420,7 @@ def prefix(request, pk):
|
|||||||
duplicate_prefix_table.exclude = ('vrf',)
|
duplicate_prefix_table.exclude = ('vrf',)
|
||||||
|
|
||||||
# Child prefixes table
|
# Child prefixes table
|
||||||
if prefix.vrf:
|
child_prefixes = Prefix.objects.filter(vrf=prefix.vrf, prefix__net_contained=str(prefix.prefix))\
|
||||||
# If the prefix is in a VRF, show child prefixes only within that VRF.
|
|
||||||
child_prefixes = Prefix.objects.filter(vrf=prefix.vrf)
|
|
||||||
else:
|
|
||||||
# If the prefix is in the global table, show child prefixes from all VRFs.
|
|
||||||
child_prefixes = Prefix.objects.all()
|
|
||||||
child_prefixes = child_prefixes.filter(prefix__net_contained=str(prefix.prefix))\
|
|
||||||
.select_related('site', 'role').annotate_depth(limit=0)
|
.select_related('site', 'role').annotate_depth(limit=0)
|
||||||
if child_prefixes:
|
if child_prefixes:
|
||||||
child_prefixes = add_available_prefixes(prefix.prefix, child_prefixes)
|
child_prefixes = add_available_prefixes(prefix.prefix, child_prefixes)
|
||||||
@ -499,7 +493,7 @@ def prefix_ipaddresses(request, pk):
|
|||||||
prefix = get_object_or_404(Prefix.objects.all(), pk=pk)
|
prefix = get_object_or_404(Prefix.objects.all(), pk=pk)
|
||||||
|
|
||||||
# Find all IPAddresses belonging to this Prefix
|
# Find all IPAddresses belonging to this Prefix
|
||||||
ipaddresses = IPAddress.objects.filter(vrf=prefix.vrf, address__net_contained_or_equal=str(prefix.prefix))\
|
ipaddresses = IPAddress.objects.filter(vrf=prefix.vrf, address__net_host_contained=str(prefix.prefix))\
|
||||||
.select_related('vrf', 'interface__device', 'primary_ip4_for', 'primary_ip6_for')
|
.select_related('vrf', 'interface__device', 'primary_ip4_for', 'primary_ip6_for')
|
||||||
ipaddresses = add_available_ipaddresses(prefix.prefix, ipaddresses, prefix.is_pool)
|
ipaddresses = add_available_ipaddresses(prefix.prefix, ipaddresses, prefix.is_pool)
|
||||||
|
|
||||||
|
@ -12,7 +12,7 @@ except ImportError:
|
|||||||
"the documentation.")
|
"the documentation.")
|
||||||
|
|
||||||
|
|
||||||
VERSION = '1.9.0-r1'
|
VERSION = '1.9.1'
|
||||||
|
|
||||||
# Import local configuration
|
# Import local configuration
|
||||||
for setting in ['ALLOWED_HOSTS', 'DATABASE', 'SECRET_KEY']:
|
for setting in ['ALLOWED_HOSTS', 'DATABASE', 'SECRET_KEY']:
|
||||||
|
@ -245,7 +245,7 @@
|
|||||||
{% if request.user.is_staff %}
|
{% if request.user.is_staff %}
|
||||||
<li><a href="{% url 'admin:index' %}"><i class="fa fa-cogs" aria-hidden="true"></i> Admin</a></li>
|
<li><a href="{% url 'admin:index' %}"><i class="fa fa-cogs" aria-hidden="true"></i> Admin</a></li>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<li><a href="{% url 'users:profile' %}"><i class="fa fa-user" aria-hidden="true"></i> Profile</a></li>
|
<li><a href="{% url 'users:profile' %}"><i class="fa fa-user" aria-hidden="true"></i> {{ request.user }}</a></li>
|
||||||
<li><a href="{% url 'logout' %}"><i class="fa fa-sign-out" aria-hidden="true"></i> Log out</a></li>
|
<li><a href="{% url 'logout' %}"><i class="fa fa-sign-out" aria-hidden="true"></i> Log out</a></li>
|
||||||
{% else %}
|
{% else %}
|
||||||
<li><a href="{% url 'login' %}?next={{ request.path }}"><i class="fa fa-sign-in" aria-hidden="true"></i> Log in</a></li>
|
<li><a href="{% url 'login' %}?next={{ request.path }}"><i class="fa fa-sign-in" aria-hidden="true"></i> Log in</a></li>
|
||||||
|
Reference in New Issue
Block a user