1
0
mirror of https://github.com/netbox-community/netbox.git synced 2024-05-10 07:54:54 +00:00

553 lines
18 KiB
Python
Raw Normal View History

from netaddr import IPSet
2016-03-01 11:23:03 -05:00
from django_tables2 import RequestConfig
2016-05-16 13:04:45 -04:00
2016-03-01 11:23:03 -05:00
from django.contrib.auth.mixins import PermissionRequiredMixin
2016-05-16 13:04:45 -04:00
from django.db.models import Count
from django.shortcuts import get_object_or_404, render
2016-03-01 11:23:03 -05:00
from dcim.models import Device
from utilities.paginator import EnhancedPaginator
2016-05-18 16:20:30 -04:00
from utilities.views import (
BulkDeleteView, BulkEditView, BulkImportView, ObjectDeleteView, ObjectEditView, ObjectListView,
)
2016-03-01 11:23:03 -05:00
2016-05-18 16:20:30 -04:00
from . import filters, forms, tables
from .models import Aggregate, IPAddress, Prefix, RIR, Role, VLAN, VRF
2016-03-01 11:23:03 -05:00
def add_available_prefixes(parent, prefix_list):
"""
Create fake Prefix objects for all unallocated space within a prefix.
"""
# Find all unallocated space
available_prefixes = IPSet(parent) ^ IPSet([p.prefix for p in prefix_list])
available_prefixes = [Prefix(prefix=p) for p in available_prefixes.iter_cidrs()]
# Concatenate and sort complete list of children
prefix_list = list(prefix_list) + available_prefixes
prefix_list.sort(key=lambda p: p.prefix)
return prefix_list
#
# VRFs
#
class VRFListView(ObjectListView):
2016-03-01 11:23:03 -05:00
queryset = VRF.objects.all()
2016-05-18 16:20:30 -04:00
filter = filters.VRFFilter
table = tables.VRFTable
edit_permissions = ['ipam.change_vrf', 'ipam.delete_vrf']
template_name = 'ipam/vrf_list.html'
2016-03-01 11:23:03 -05:00
def vrf(request, pk):
vrf = get_object_or_404(VRF.objects.all(), pk=pk)
prefixes = Prefix.objects.filter(vrf=vrf)
return render(request, 'ipam/vrf.html', {
'vrf': vrf,
'prefixes': prefixes,
})
class VRFEditView(PermissionRequiredMixin, ObjectEditView):
permission_required = 'ipam.change_vrf'
model = VRF
2016-05-18 16:20:30 -04:00
form_class = forms.VRFForm
cancel_url = 'ipam:vrf_list'
2016-03-01 11:23:03 -05:00
class VRFDeleteView(PermissionRequiredMixin, ObjectDeleteView):
permission_required = 'ipam.delete_vrf'
model = VRF
redirect_url = 'ipam:vrf_list'
2016-03-01 11:23:03 -05:00
class VRFBulkImportView(PermissionRequiredMixin, BulkImportView):
permission_required = 'ipam.add_vrf'
2016-05-18 16:20:30 -04:00
form = forms.VRFImportForm
table = tables.VRFTable
2016-03-01 11:23:03 -05:00
template_name = 'ipam/vrf_import.html'
obj_list_url = 'ipam:vrf_list'
class VRFBulkEditView(PermissionRequiredMixin, BulkEditView):
permission_required = 'ipam.change_vrf'
cls = VRF
2016-05-18 16:20:30 -04:00
form = forms.VRFBulkEditForm
2016-03-01 11:23:03 -05:00
template_name = 'ipam/vrf_bulk_edit.html'
default_redirect_url = 'ipam:vrf_list'
2016-03-01 11:23:03 -05:00
def update_objects(self, pk_list, form):
fields_to_update = {}
for field in ['description']:
if form.cleaned_data[field]:
fields_to_update[field] = form.cleaned_data[field]
return self.cls.objects.filter(pk__in=pk_list).update(**fields_to_update)
2016-03-01 11:23:03 -05:00
class VRFBulkDeleteView(PermissionRequiredMixin, BulkDeleteView):
permission_required = 'ipam.delete_vrf'
cls = VRF
2016-05-18 16:20:30 -04:00
form = forms.VRFBulkDeleteForm
default_redirect_url = 'ipam:vrf_list'
2016-03-01 11:23:03 -05:00
2016-05-16 13:04:45 -04:00
#
# RIRs
#
class RIRListView(ObjectListView):
queryset = RIR.objects.annotate(aggregate_count=Count('aggregates'))
2016-05-18 16:20:30 -04:00
table = tables.RIRTable
2016-05-16 13:04:45 -04:00
edit_permissions = ['ipam.change_rir', 'ipam.delete_rir']
template_name = 'ipam/rir_list.html'
class RIREditView(PermissionRequiredMixin, ObjectEditView):
permission_required = 'ipam.change_rir'
model = RIR
2016-05-18 16:20:30 -04:00
form_class = forms.RIRForm
2016-05-16 13:04:45 -04:00
success_url = 'ipam:rir_list'
cancel_url = 'ipam:rir_list'
class RIRBulkDeleteView(PermissionRequiredMixin, BulkDeleteView):
permission_required = 'ipam.delete_rir'
cls = RIR
2016-05-18 16:20:30 -04:00
form = forms.RIRBulkDeleteForm
2016-05-16 13:04:45 -04:00
default_redirect_url = 'ipam:rir_list'
2016-03-01 11:23:03 -05:00
#
# Aggregates
#
class AggregateListView(ObjectListView):
queryset = Aggregate.objects.select_related('rir').extra(select={
'child_count': 'SELECT COUNT(*) FROM ipam_prefix WHERE ipam_prefix.prefix <<= ipam_aggregate.prefix',
2016-03-01 11:23:03 -05:00
})
2016-05-18 16:20:30 -04:00
filter = filters.AggregateFilter
filter_form = forms.AggregateFilterForm
table = tables.AggregateTable
edit_permissions = ['ipam.change_aggregate', 'ipam.delete_aggregate']
template_name = 'ipam/aggregate_list.html'
2016-03-01 11:23:03 -05:00
def extra_context(self):
ipv4_total = 0
ipv6_total = 0
for a in self.queryset:
if a.prefix.version == 4:
ipv4_total += a.prefix.size
elif a.prefix.version == 6:
ipv6_total += a.prefix.size / 2**64
return {
'ipv4_total': ipv4_total,
'ipv6_total': ipv6_total,
}
2016-03-01 11:23:03 -05:00
def aggregate(request, pk):
aggregate = get_object_or_404(Aggregate, pk=pk)
# Find all child prefixes contained by this aggregate
child_prefixes = Prefix.objects.filter(prefix__net_contained_or_equal=str(aggregate.prefix))\
.select_related('site', 'role').annotate_depth(limit=0)
2016-03-01 11:23:03 -05:00
child_prefixes = add_available_prefixes(aggregate.prefix, child_prefixes)
2016-05-18 16:20:30 -04:00
prefix_table = tables.PrefixTable(child_prefixes)
prefix_table.model = Prefix
2016-03-01 11:23:03 -05:00
if request.user.has_perm('ipam.change_prefix') or request.user.has_perm('ipam.delete_prefix'):
prefix_table.base_columns['pk'].visible = True
RequestConfig(request, paginate={'klass': EnhancedPaginator}).configure(prefix_table)
2016-03-01 11:23:03 -05:00
return render(request, 'ipam/aggregate.html', {
'aggregate': aggregate,
'prefix_table': prefix_table,
})
class AggregateEditView(PermissionRequiredMixin, ObjectEditView):
permission_required = 'ipam.change_aggregate'
model = Aggregate
2016-05-18 16:20:30 -04:00
form_class = forms.AggregateForm
cancel_url = 'ipam:aggregate_list'
2016-03-01 11:23:03 -05:00
class AggregateDeleteView(PermissionRequiredMixin, ObjectDeleteView):
permission_required = 'ipam.delete_aggregate'
model = Aggregate
redirect_url = 'ipam:aggregate_list'
2016-03-01 11:23:03 -05:00
class AggregateBulkImportView(PermissionRequiredMixin, BulkImportView):
permission_required = 'ipam.add_aggregate'
2016-05-18 16:20:30 -04:00
form = forms.AggregateImportForm
table = tables.AggregateTable
2016-03-01 11:23:03 -05:00
template_name = 'ipam/aggregate_import.html'
obj_list_url = 'ipam:aggregate_list'
class AggregateBulkEditView(PermissionRequiredMixin, BulkEditView):
permission_required = 'ipam.change_aggregate'
cls = Aggregate
2016-05-18 16:20:30 -04:00
form = forms.AggregateBulkEditForm
2016-03-01 11:23:03 -05:00
template_name = 'ipam/aggregate_bulk_edit.html'
default_redirect_url = 'ipam:aggregate_list'
2016-03-01 11:23:03 -05:00
def update_objects(self, pk_list, form):
fields_to_update = {}
for field in ['rir', 'date_added', 'description']:
if form.cleaned_data[field]:
fields_to_update[field] = form.cleaned_data[field]
return self.cls.objects.filter(pk__in=pk_list).update(**fields_to_update)
2016-03-01 11:23:03 -05:00
class AggregateBulkDeleteView(PermissionRequiredMixin, BulkDeleteView):
permission_required = 'ipam.delete_aggregate'
cls = Aggregate
2016-05-18 16:20:30 -04:00
form = forms.AggregateBulkDeleteForm
default_redirect_url = 'ipam:aggregate_list'
2016-03-01 11:23:03 -05:00
2016-05-17 15:04:16 -04:00
#
# Prefix/VLAN roles
#
class RoleListView(ObjectListView):
queryset = Role.objects.all()
2016-05-18 16:20:30 -04:00
table = tables.RoleTable
2016-05-17 15:04:16 -04:00
edit_permissions = ['ipam.change_role', 'ipam.delete_role']
template_name = 'ipam/role_list.html'
class RoleEditView(PermissionRequiredMixin, ObjectEditView):
permission_required = 'ipam.change_role'
model = Role
2016-05-18 16:20:30 -04:00
form_class = forms.RoleForm
2016-05-17 15:04:16 -04:00
success_url = 'ipam:role_list'
cancel_url = 'ipam:role_list'
class RoleBulkDeleteView(PermissionRequiredMixin, BulkDeleteView):
permission_required = 'ipam.delete_role'
cls = Role
2016-05-18 16:20:30 -04:00
form = forms.RoleBulkDeleteForm
2016-05-17 15:04:16 -04:00
default_redirect_url = 'ipam:role_list'
2016-03-01 11:23:03 -05:00
#
# Prefixes
#
class PrefixListView(ObjectListView):
queryset = Prefix.objects.select_related('site', 'role')
2016-05-18 16:20:30 -04:00
filter = filters.PrefixFilter
filter_form = forms.PrefixFilterForm
table = tables.PrefixTable
edit_permissions = ['ipam.change_prefix', 'ipam.delete_prefix']
template_name = 'ipam/prefix_list.html'
2016-03-01 11:23:03 -05:00
2016-03-07 12:56:37 -05:00
def alter_queryset(self, request):
# Show only top-level prefixes by default (unless searching)
limit = None if request.GET.get('expand') or request.GET.get('q') else 0
2016-03-07 12:56:37 -05:00
return self.queryset.annotate_depth(limit=limit)
2016-03-01 11:23:03 -05:00
def prefix(request, pk):
prefix = get_object_or_404(Prefix.objects.select_related('site', 'vlan', 'role'), pk=pk)
2016-03-01 11:23:03 -05:00
try:
aggregate = Aggregate.objects.get(prefix__net_contains_or_equals=str(prefix.prefix))
except Aggregate.DoesNotExist:
aggregate = None
# Count child IP addresses
ipaddress_count = IPAddress.objects.filter(address__net_contained_or_equal=str(prefix.prefix)).count()
# Parent prefixes table
parent_prefixes = Prefix.objects.filter(vrf=prefix.vrf, prefix__net_contains=str(prefix.prefix))\
.select_related('site', 'role').annotate_depth()
2016-05-18 16:20:30 -04:00
parent_prefix_table = tables.PrefixBriefTable(parent_prefixes)
2016-03-01 11:23:03 -05:00
# Duplicate prefixes table
duplicate_prefixes = Prefix.objects.filter(vrf=prefix.vrf, prefix=str(prefix.prefix)).exclude(pk=prefix.pk)\
.select_related('site', 'role')
2016-05-18 16:20:30 -04:00
duplicate_prefix_table = tables.PrefixBriefTable(duplicate_prefixes)
2016-03-01 11:23:03 -05:00
# Child prefixes table
child_prefixes = Prefix.objects.filter(vrf=prefix.vrf, prefix__net_contained=str(prefix.prefix))\
.select_related('site', 'role').annotate_depth(limit=0)
2016-03-01 11:23:03 -05:00
if child_prefixes:
child_prefixes = add_available_prefixes(prefix.prefix, child_prefixes)
2016-05-18 16:20:30 -04:00
child_prefix_table = tables.PrefixTable(child_prefixes)
child_prefix_table.model = Prefix
2016-03-01 11:23:03 -05:00
if request.user.has_perm('ipam.change_prefix') or request.user.has_perm('ipam.delete_prefix'):
child_prefix_table.base_columns['pk'].visible = True
RequestConfig(request, paginate={'klass': EnhancedPaginator}).configure(child_prefix_table)
2016-03-01 11:23:03 -05:00
return render(request, 'ipam/prefix.html', {
'prefix': prefix,
'aggregate': aggregate,
'ipaddress_count': ipaddress_count,
'parent_prefix_table': parent_prefix_table,
'child_prefix_table': child_prefix_table,
'duplicate_prefix_table': duplicate_prefix_table,
})
class PrefixEditView(PermissionRequiredMixin, ObjectEditView):
permission_required = 'ipam.change_prefix'
model = Prefix
2016-05-18 16:20:30 -04:00
form_class = forms.PrefixForm
fields_initial = ['site', 'vrf', 'prefix']
cancel_url = 'ipam:prefix_list'
2016-03-01 11:23:03 -05:00
class PrefixDeleteView(PermissionRequiredMixin, ObjectDeleteView):
permission_required = 'ipam.delete_prefix'
model = Prefix
redirect_url = 'ipam:prefix_list'
2016-03-01 11:23:03 -05:00
class PrefixBulkImportView(PermissionRequiredMixin, BulkImportView):
permission_required = 'ipam.add_prefix'
2016-05-18 16:20:30 -04:00
form = forms.PrefixImportForm
table = tables.PrefixTable
2016-03-01 11:23:03 -05:00
template_name = 'ipam/prefix_import.html'
obj_list_url = 'ipam:prefix_list'
class PrefixBulkEditView(PermissionRequiredMixin, BulkEditView):
permission_required = 'ipam.change_prefix'
cls = Prefix
2016-05-18 16:20:30 -04:00
form = forms.PrefixBulkEditForm
2016-03-01 11:23:03 -05:00
template_name = 'ipam/prefix_bulk_edit.html'
default_redirect_url = 'ipam:prefix_list'
2016-03-01 11:23:03 -05:00
def update_objects(self, pk_list, form):
fields_to_update = {}
if form.cleaned_data['vrf']:
fields_to_update['vrf'] = form.cleaned_data['vrf']
elif form.cleaned_data['vrf_global']:
fields_to_update['vrf'] = None
for field in ['site', 'status', 'role', 'description']:
if form.cleaned_data[field]:
fields_to_update[field] = form.cleaned_data[field]
return self.cls.objects.filter(pk__in=pk_list).update(**fields_to_update)
2016-03-01 11:23:03 -05:00
class PrefixBulkDeleteView(PermissionRequiredMixin, BulkDeleteView):
permission_required = 'ipam.delete_prefix'
cls = Prefix
2016-05-18 16:20:30 -04:00
form = forms.PrefixBulkDeleteForm
default_redirect_url = 'ipam:prefix_list'
2016-03-01 11:23:03 -05:00
def prefix_ipaddresses(request, pk):
prefix = get_object_or_404(Prefix.objects.all(), pk=pk)
# Find all IPAddresses belonging to this Prefix
ipaddresses = IPAddress.objects.filter(address__net_contained_or_equal=str(prefix.prefix))\
.select_related('vrf', 'interface__device', 'primary_ip4_for', 'primary_ip6_for')
2016-03-01 11:23:03 -05:00
2016-05-18 16:20:30 -04:00
ip_table = tables.IPAddressTable(ipaddresses)
ip_table.model = IPAddress
2016-03-01 11:23:03 -05:00
if request.user.has_perm('ipam.change_ipaddress') or request.user.has_perm('ipam.delete_ipaddress'):
ip_table.base_columns['pk'].visible = True
RequestConfig(request, paginate={'klass': EnhancedPaginator}).configure(ip_table)
2016-03-01 11:23:03 -05:00
return render(request, 'ipam/prefix_ipaddresses.html', {
'prefix': prefix,
'ip_table': ip_table,
})
#
# IP addresses
#
class IPAddressListView(ObjectListView):
queryset = IPAddress.objects.select_related('vrf', 'interface__device', 'primary_ip4_for', 'primary_ip6_for')
2016-05-18 16:20:30 -04:00
filter = filters.IPAddressFilter
filter_form = forms.IPAddressFilterForm
table = tables.IPAddressTable
edit_permissions = ['ipam.change_ipaddress', 'ipam.delete_ipaddress']
template_name = 'ipam/ipaddress_list.html'
2016-03-01 11:23:03 -05:00
def ipaddress(request, pk):
ipaddress = get_object_or_404(IPAddress.objects.select_related('interface__device'), pk=pk)
# Parent prefixes table
2016-03-01 11:23:03 -05:00
parent_prefixes = Prefix.objects.filter(vrf=ipaddress.vrf, prefix__net_contains=str(ipaddress.address.ip))
parent_prefixes_table = tables.PrefixBriefTable(parent_prefixes)
2016-03-01 11:23:03 -05:00
# Duplicate IPs table
duplicate_ips = IPAddress.objects.filter(vrf=ipaddress.vrf, address=str(ipaddress.address))\
.exclude(pk=ipaddress.pk).select_related('interface__device', 'nat_inside')
duplicate_ips_table = tables.IPAddressBriefTable(duplicate_ips)
# Related IP table
related_ips = IPAddress.objects.select_related('interface__device').exclude(address=str(ipaddress.address))\
.filter(vrf=ipaddress.vrf, address__net_contained_or_equal=str(ipaddress.address))
2016-05-18 16:20:30 -04:00
related_ips_table = tables.IPAddressBriefTable(related_ips)
2016-03-01 11:23:03 -05:00
return render(request, 'ipam/ipaddress.html', {
'ipaddress': ipaddress,
'parent_prefixes_table': parent_prefixes_table,
'duplicate_ips_table': duplicate_ips_table,
2016-03-01 11:23:03 -05:00
'related_ips_table': related_ips_table,
})
class IPAddressEditView(PermissionRequiredMixin, ObjectEditView):
permission_required = 'ipam.change_ipaddress'
model = IPAddress
2016-05-18 16:20:30 -04:00
form_class = forms.IPAddressForm
fields_initial = ['address', 'vrf']
template_name = 'ipam/ipaddress_edit.html'
cancel_url = 'ipam:ipaddress_list'
2016-03-01 11:23:03 -05:00
class IPAddressDeleteView(PermissionRequiredMixin, ObjectDeleteView):
permission_required = 'ipam.delete_ipaddress'
model = IPAddress
redirect_url = 'ipam:ipaddress_list'
2016-03-01 11:23:03 -05:00
class IPAddressBulkImportView(PermissionRequiredMixin, BulkImportView):
permission_required = 'ipam.add_ipaddress'
2016-05-18 16:20:30 -04:00
form = forms.IPAddressImportForm
table = tables.IPAddressTable
2016-03-01 11:23:03 -05:00
template_name = 'ipam/ipaddress_import.html'
obj_list_url = 'ipam:ipaddress_list'
def save_obj(self, obj):
obj.save()
# Update primary IP for device if needed
try:
if obj.family == 4 and obj.primary_ip4_for:
device = obj.primary_ip4_for
device.primary_ip4 = obj
device.save()
elif obj.family == 6 and obj.primary_ip6_for:
device = obj.primary_ip6_for
device.primary_ip6 = obj
device.save()
2016-03-01 11:23:03 -05:00
except Device.DoesNotExist:
pass
class IPAddressBulkEditView(PermissionRequiredMixin, BulkEditView):
permission_required = 'ipam.change_ipaddress'
cls = IPAddress
2016-05-18 16:20:30 -04:00
form = forms.IPAddressBulkEditForm
2016-03-01 11:23:03 -05:00
template_name = 'ipam/ipaddress_bulk_edit.html'
default_redirect_url = 'ipam:ipaddress_list'
2016-03-01 11:23:03 -05:00
def update_objects(self, pk_list, form):
fields_to_update = {}
if form.cleaned_data['vrf']:
fields_to_update['vrf'] = form.cleaned_data['vrf']
elif form.cleaned_data['vrf_global']:
fields_to_update['vrf'] = None
for field in ['description']:
if form.cleaned_data[field]:
fields_to_update[field] = form.cleaned_data[field]
return self.cls.objects.filter(pk__in=pk_list).update(**fields_to_update)
2016-03-01 11:23:03 -05:00
class IPAddressBulkDeleteView(PermissionRequiredMixin, BulkDeleteView):
permission_required = 'ipam.delete_ipaddress'
cls = IPAddress
2016-05-18 16:20:30 -04:00
form = forms.IPAddressBulkDeleteForm
default_redirect_url = 'ipam:ipaddress_list'
2016-03-01 11:23:03 -05:00
#
# VLANs
#
class VLANListView(ObjectListView):
queryset = VLAN.objects.select_related('site', 'role')
2016-05-18 16:20:30 -04:00
filter = filters.VLANFilter
filter_form = forms.VLANFilterForm
table = tables.VLANTable
edit_permissions = ['ipam.change_vlan', 'ipam.delete_vlan']
template_name = 'ipam/vlan_list.html'
2016-03-01 11:23:03 -05:00
def vlan(request, pk):
vlan = get_object_or_404(VLAN.objects.select_related('site', 'role'), pk=pk)
2016-03-01 11:23:03 -05:00
prefixes = Prefix.objects.filter(vlan=vlan)
return render(request, 'ipam/vlan.html', {
'vlan': vlan,
'prefixes': prefixes,
})
class VLANEditView(PermissionRequiredMixin, ObjectEditView):
permission_required = 'ipam.change_vlan'
model = VLAN
2016-05-18 16:20:30 -04:00
form_class = forms.VLANForm
cancel_url = 'ipam:vlan_list'
2016-03-01 11:23:03 -05:00
class VLANDeleteView(PermissionRequiredMixin, ObjectDeleteView):
permission_required = 'ipam.delete_vlan'
model = VLAN
redirect_url = 'ipam:vlan_list'
2016-03-01 11:23:03 -05:00
class VLANBulkImportView(PermissionRequiredMixin, BulkImportView):
permission_required = 'ipam.add_vlan'
2016-05-18 16:20:30 -04:00
form = forms.VLANImportForm
table = tables.VLANTable
2016-03-01 11:23:03 -05:00
template_name = 'ipam/vlan_import.html'
obj_list_url = 'ipam:vlan_list'
class VLANBulkEditView(PermissionRequiredMixin, BulkEditView):
permission_required = 'ipam.change_vlan'
cls = VLAN
2016-05-18 16:20:30 -04:00
form = forms.VLANBulkEditForm
2016-03-01 11:23:03 -05:00
template_name = 'ipam/vlan_bulk_edit.html'
default_redirect_url = 'ipam:vlan_list'
2016-03-01 11:23:03 -05:00
def update_objects(self, pk_list, form):
fields_to_update = {}
for field in ['site', 'status', 'role']:
if form.cleaned_data[field]:
fields_to_update[field] = form.cleaned_data[field]
return self.cls.objects.filter(pk__in=pk_list).update(**fields_to_update)
2016-03-01 11:23:03 -05:00
class VLANBulkDeleteView(PermissionRequiredMixin, BulkDeleteView):
permission_required = 'ipam.delete_vlan'
cls = VLAN
2016-05-18 16:20:30 -04:00
form = forms.VLANBulkDeleteForm
default_redirect_url = 'ipam:vlan_list'