mirror of
https://github.com/netbox-community/netbox.git
synced 2024-05-10 07:54:54 +00:00
cef0d168a5
* netbox-community/netbox#6930: Add ID column to devices, device types, and components * netbox-community/netbox#6930: Add ID column to sites, racks, and tenants * netbox-community/netbox#6930: Add ID column to power, providers, TODO circuits * netbox-community/netbox#6930: Add ID column to virtualization tables * netbox-community/netbox#6930: Add ID column to IPAM tables * netbox-community/netbox#6930: Add ID column to 'extras' tables * netbox-community/netbox#6930: Move ID column to BaseTable class * netbox-community/netbox#6930: Don't linkify ID in device component template tables * netbox-community/netbox#6930: Don't show ID column in interface/console/power connections tables * netbox-community/netbox#6930: Don't show ID column in device component template tables * netbox-community/netbox#6930: Add ID column to ObjectJournal, DeviceImport, and Circuit tables * Exclude ID column from selected tables * netbox-community/netbox#6930:revert default columns on ObjectChangeTable, not configurable * netbox-community/netbox#6930: Add object ID to tagged objects table in tag detail view Co-authored-by: Jeremy Stretch <jstretch@ns1.com>
379 lines
10 KiB
Python
379 lines
10 KiB
Python
import django_tables2 as tables
|
|
from django.utils.safestring import mark_safe
|
|
from django_tables2.utils import Accessor
|
|
|
|
from tenancy.tables import TenantColumn
|
|
from utilities.tables import (
|
|
BaseTable, BooleanColumn, ButtonsColumn, ChoiceFieldColumn, LinkedCountColumn, TagColumn,
|
|
ToggleColumn, UtilizationColumn,
|
|
)
|
|
from ipam.models import *
|
|
|
|
__all__ = (
|
|
'AggregateTable',
|
|
'InterfaceIPAddressTable',
|
|
'IPAddressAssignTable',
|
|
'IPAddressTable',
|
|
'IPRangeTable',
|
|
'PrefixTable',
|
|
'RIRTable',
|
|
'RoleTable',
|
|
)
|
|
|
|
AVAILABLE_LABEL = mark_safe('<span class="badge bg-success">Available</span>')
|
|
|
|
PREFIX_LINK = """
|
|
{% load helpers %}
|
|
{% if record.depth %}
|
|
<div class="record-depth">
|
|
{% for i in record.depth|as_range %}
|
|
<span>•</span>
|
|
{% endfor %}
|
|
</div>
|
|
{% endif %}
|
|
<a href="{% if record.pk %}{% url 'ipam:prefix' pk=record.pk %}{% else %}{% url 'ipam:prefix_add' %}?prefix={{ record }}{% if object.vrf %}&vrf={{ object.vrf.pk }}{% endif %}{% if object.site %}&site={{ object.site.pk }}{% endif %}{% if object.tenant %}&tenant_group={{ object.tenant.group.pk }}&tenant={{ object.tenant.pk }}{% endif %}{% endif %}">{{ record.prefix }}</a>
|
|
"""
|
|
|
|
PREFIXFLAT_LINK = """
|
|
{% load helpers %}
|
|
{% if record.pk %}
|
|
<a href="{% url 'ipam:prefix' pk=record.pk %}">{{ record.prefix }}</a>
|
|
{% else %}
|
|
{{ record.prefix }}
|
|
{% endif %}
|
|
"""
|
|
|
|
IPADDRESS_LINK = """
|
|
{% if record.pk %}
|
|
<a href="{{ record.get_absolute_url }}">{{ record.address }}</a>
|
|
{% elif perms.ipam.add_ipaddress %}
|
|
<a href="{% url 'ipam:ipaddress_add' %}?address={{ record.1 }}{% if object.vrf %}&vrf={{ object.vrf.pk }}{% endif %}{% if object.tenant %}&tenant={{ object.tenant.pk }}{% endif %}" class="btn btn-sm btn-success">{% if record.0 <= 65536 %}{{ record.0 }}{% else %}Many{% endif %} IP{{ record.0|pluralize }} available</a>
|
|
{% else %}
|
|
{% if record.0 <= 65536 %}{{ record.0 }}{% else %}Many{% endif %} IP{{ record.0|pluralize }} available
|
|
{% endif %}
|
|
"""
|
|
|
|
IPADDRESS_ASSIGN_LINK = """
|
|
<a href="{% url 'ipam:ipaddress_edit' pk=record.pk %}?{% if request.GET.interface %}interface={{ request.GET.interface }}{% elif request.GET.vminterface %}vminterface={{ request.GET.vminterface }}{% endif %}&return_url={{ request.GET.return_url }}">{{ record }}</a>
|
|
"""
|
|
|
|
VRF_LINK = """
|
|
{% if record.vrf %}
|
|
<a href="{{ record.vrf.get_absolute_url }}">{{ record.vrf }}</a>
|
|
{% elif object.vrf %}
|
|
<a href="{{ object.vrf.get_absolute_url }}">{{ object.vrf }}</a>
|
|
{% else %}
|
|
Global
|
|
{% endif %}
|
|
"""
|
|
|
|
|
|
#
|
|
# RIRs
|
|
#
|
|
|
|
class RIRTable(BaseTable):
|
|
pk = ToggleColumn()
|
|
name = tables.Column(
|
|
linkify=True
|
|
)
|
|
is_private = BooleanColumn(
|
|
verbose_name='Private'
|
|
)
|
|
aggregate_count = LinkedCountColumn(
|
|
viewname='ipam:aggregate_list',
|
|
url_params={'rir_id': 'pk'},
|
|
verbose_name='Aggregates'
|
|
)
|
|
actions = ButtonsColumn(RIR)
|
|
|
|
class Meta(BaseTable.Meta):
|
|
model = RIR
|
|
fields = ('pk', 'id', 'name', 'slug', 'is_private', 'aggregate_count', 'description', 'actions')
|
|
default_columns = ('pk', 'name', 'is_private', 'aggregate_count', 'description', 'actions')
|
|
|
|
|
|
#
|
|
# Aggregates
|
|
#
|
|
|
|
class AggregateTable(BaseTable):
|
|
pk = ToggleColumn()
|
|
prefix = tables.Column(
|
|
linkify=True,
|
|
verbose_name='Aggregate'
|
|
)
|
|
tenant = TenantColumn()
|
|
date_added = tables.DateColumn(
|
|
format="Y-m-d",
|
|
verbose_name='Added'
|
|
)
|
|
child_count = tables.Column(
|
|
verbose_name='Prefixes'
|
|
)
|
|
utilization = UtilizationColumn(
|
|
accessor='get_utilization',
|
|
orderable=False
|
|
)
|
|
tags = TagColumn(
|
|
url_name='ipam:aggregate_list'
|
|
)
|
|
|
|
class Meta(BaseTable.Meta):
|
|
model = Aggregate
|
|
fields = ('pk', 'id', 'prefix', 'rir', 'tenant', 'child_count', 'utilization', 'date_added', 'description', 'tags')
|
|
default_columns = ('pk', 'prefix', 'rir', 'tenant', 'child_count', 'utilization', 'date_added', 'description')
|
|
|
|
|
|
#
|
|
# Roles
|
|
#
|
|
|
|
class RoleTable(BaseTable):
|
|
pk = ToggleColumn()
|
|
name = tables.Column(
|
|
linkify=True
|
|
)
|
|
prefix_count = LinkedCountColumn(
|
|
viewname='ipam:prefix_list',
|
|
url_params={'role_id': 'pk'},
|
|
verbose_name='Prefixes'
|
|
)
|
|
vlan_count = LinkedCountColumn(
|
|
viewname='ipam:vlan_list',
|
|
url_params={'role_id': 'pk'},
|
|
verbose_name='VLANs'
|
|
)
|
|
actions = ButtonsColumn(Role)
|
|
|
|
class Meta(BaseTable.Meta):
|
|
model = Role
|
|
fields = ('pk', 'id', 'name', 'slug', 'prefix_count', 'vlan_count', 'description', 'weight', 'actions')
|
|
default_columns = ('pk', 'name', 'prefix_count', 'vlan_count', 'description', 'actions')
|
|
|
|
|
|
#
|
|
# Prefixes
|
|
#
|
|
|
|
class PrefixUtilizationColumn(UtilizationColumn):
|
|
"""
|
|
Extend UtilizationColumn to allow disabling the warning & danger thresholds for prefixes
|
|
marked as fully utilized.
|
|
"""
|
|
template_code = """
|
|
{% load helpers %}
|
|
{% if record.pk and record.mark_utilized %}
|
|
{% utilization_graph value warning_threshold=0 danger_threshold=0 %}
|
|
{% elif record.pk %}
|
|
{% utilization_graph value %}
|
|
{% endif %}
|
|
"""
|
|
|
|
|
|
class PrefixTable(BaseTable):
|
|
pk = ToggleColumn()
|
|
prefix = tables.TemplateColumn(
|
|
template_code=PREFIX_LINK,
|
|
attrs={'td': {'class': 'text-nowrap'}}
|
|
)
|
|
prefix_flat = tables.TemplateColumn(
|
|
template_code=PREFIXFLAT_LINK,
|
|
attrs={'td': {'class': 'text-nowrap'}},
|
|
verbose_name='Prefix (Flat)',
|
|
)
|
|
depth = tables.Column(
|
|
accessor=Accessor('_depth'),
|
|
verbose_name='Depth'
|
|
)
|
|
children = LinkedCountColumn(
|
|
accessor=Accessor('_children'),
|
|
viewname='ipam:prefix_list',
|
|
url_params={
|
|
'vrf_id': 'vrf_id',
|
|
'within': 'prefix',
|
|
},
|
|
verbose_name='Children'
|
|
)
|
|
status = ChoiceFieldColumn(
|
|
default=AVAILABLE_LABEL
|
|
)
|
|
vrf = tables.TemplateColumn(
|
|
template_code=VRF_LINK,
|
|
verbose_name='VRF'
|
|
)
|
|
tenant = TenantColumn()
|
|
site = tables.Column(
|
|
linkify=True
|
|
)
|
|
vlan = tables.Column(
|
|
linkify=True,
|
|
verbose_name='VLAN'
|
|
)
|
|
role = tables.Column(
|
|
linkify=True
|
|
)
|
|
is_pool = BooleanColumn(
|
|
verbose_name='Pool'
|
|
)
|
|
mark_utilized = BooleanColumn(
|
|
verbose_name='Marked Utilized'
|
|
)
|
|
utilization = PrefixUtilizationColumn(
|
|
accessor='get_utilization',
|
|
orderable=False
|
|
)
|
|
tags = TagColumn(
|
|
url_name='ipam:prefix_list'
|
|
)
|
|
|
|
class Meta(BaseTable.Meta):
|
|
model = Prefix
|
|
fields = (
|
|
'pk', 'id', 'prefix', 'prefix_flat', 'status', 'children', 'vrf', 'utilization', 'tenant', 'site', 'vlan', 'role',
|
|
'is_pool', 'mark_utilized', 'description', 'tags',
|
|
)
|
|
default_columns = (
|
|
'pk', 'prefix', 'status', 'children', 'vrf', 'utilization', 'tenant', 'site', 'vlan', 'role', 'description',
|
|
)
|
|
row_attrs = {
|
|
'class': lambda record: 'success' if not record.pk else '',
|
|
}
|
|
|
|
|
|
#
|
|
# IP ranges
|
|
#
|
|
class IPRangeTable(BaseTable):
|
|
pk = ToggleColumn()
|
|
start_address = tables.Column(
|
|
linkify=True
|
|
)
|
|
vrf = tables.TemplateColumn(
|
|
template_code=VRF_LINK,
|
|
verbose_name='VRF'
|
|
)
|
|
status = ChoiceFieldColumn(
|
|
default=AVAILABLE_LABEL
|
|
)
|
|
role = tables.Column(
|
|
linkify=True
|
|
)
|
|
tenant = TenantColumn()
|
|
utilization = UtilizationColumn(
|
|
accessor='utilization',
|
|
orderable=False
|
|
)
|
|
|
|
class Meta(BaseTable.Meta):
|
|
model = IPRange
|
|
fields = (
|
|
'pk', 'id', 'start_address', 'end_address', 'size', 'vrf', 'status', 'role', 'tenant', 'description',
|
|
'utilization', 'tags',
|
|
)
|
|
default_columns = (
|
|
'pk', 'start_address', 'end_address', 'size', 'vrf', 'status', 'role', 'tenant', 'description',
|
|
)
|
|
row_attrs = {
|
|
'class': lambda record: 'success' if not record.pk else '',
|
|
}
|
|
|
|
|
|
#
|
|
# IPAddresses
|
|
#
|
|
|
|
class IPAddressTable(BaseTable):
|
|
pk = ToggleColumn()
|
|
address = tables.TemplateColumn(
|
|
template_code=IPADDRESS_LINK,
|
|
verbose_name='IP Address'
|
|
)
|
|
vrf = tables.TemplateColumn(
|
|
template_code=VRF_LINK,
|
|
verbose_name='VRF'
|
|
)
|
|
status = ChoiceFieldColumn(
|
|
default=AVAILABLE_LABEL
|
|
)
|
|
role = ChoiceFieldColumn()
|
|
tenant = TenantColumn()
|
|
assigned_object = tables.Column(
|
|
linkify=True,
|
|
orderable=False,
|
|
verbose_name='Interface'
|
|
)
|
|
assigned_object_parent = tables.Column(
|
|
accessor='assigned_object.parent_object',
|
|
linkify=True,
|
|
orderable=False,
|
|
verbose_name='Device/VM'
|
|
)
|
|
nat_inside = tables.Column(
|
|
linkify=True,
|
|
orderable=False,
|
|
verbose_name='NAT (Inside)'
|
|
)
|
|
assigned = BooleanColumn(
|
|
accessor='assigned_object',
|
|
linkify=True,
|
|
verbose_name='Assigned'
|
|
)
|
|
tags = TagColumn(
|
|
url_name='ipam:ipaddress_list'
|
|
)
|
|
|
|
class Meta(BaseTable.Meta):
|
|
model = IPAddress
|
|
fields = (
|
|
'pk', 'id', 'address', 'vrf', 'status', 'role', 'tenant', 'nat_inside', 'assigned', 'dns_name', 'description',
|
|
'tags',
|
|
)
|
|
default_columns = (
|
|
'pk', 'address', 'vrf', 'status', 'role', 'tenant', 'assigned', 'dns_name', 'description',
|
|
)
|
|
row_attrs = {
|
|
'class': lambda record: 'success' if not isinstance(record, IPAddress) else '',
|
|
}
|
|
|
|
|
|
class IPAddressAssignTable(BaseTable):
|
|
address = tables.TemplateColumn(
|
|
template_code=IPADDRESS_ASSIGN_LINK,
|
|
verbose_name='IP Address'
|
|
)
|
|
status = ChoiceFieldColumn()
|
|
assigned_object = tables.Column(
|
|
orderable=False
|
|
)
|
|
|
|
class Meta(BaseTable.Meta):
|
|
model = IPAddress
|
|
fields = ('address', 'dns_name', 'vrf', 'status', 'role', 'tenant', 'assigned_object', 'description')
|
|
exclude = ('id', )
|
|
orderable = False
|
|
|
|
|
|
class InterfaceIPAddressTable(BaseTable):
|
|
"""
|
|
List IP addresses assigned to a specific Interface.
|
|
"""
|
|
address = tables.Column(
|
|
linkify=True,
|
|
verbose_name='IP Address'
|
|
)
|
|
vrf = tables.TemplateColumn(
|
|
template_code=VRF_LINK,
|
|
verbose_name='VRF'
|
|
)
|
|
status = ChoiceFieldColumn()
|
|
tenant = TenantColumn()
|
|
actions = ButtonsColumn(
|
|
model=IPAddress
|
|
)
|
|
|
|
class Meta(BaseTable.Meta):
|
|
model = IPAddress
|
|
fields = ('address', 'vrf', 'status', 'role', 'tenant', 'description')
|
|
exclude = ('id', )
|