2021-03-18 11:57:59 -04:00
|
|
|
from django.core.exceptions import ValidationError
|
2016-03-01 11:23:03 -05:00
|
|
|
from django.db import models
|
2017-04-05 14:40:25 -04:00
|
|
|
from django.urls import reverse
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2016-07-13 15:30:15 -04:00
|
|
|
from dcim.fields import ASNField
|
2020-10-01 13:05:00 -04:00
|
|
|
from dcim.models import CableTermination, PathEndpoint
|
2021-03-10 14:32:50 -05:00
|
|
|
from extras.models import ObjectChange
|
2020-03-14 03:03:22 -04:00
|
|
|
from extras.utils import extras_features
|
2021-03-10 13:35:13 -05:00
|
|
|
from netbox.models import BigIDModel, ChangeLoggedModel, OrganizationalModel, PrimaryModel
|
2020-05-29 16:27:36 -04:00
|
|
|
from utilities.querysets import RestrictedQuerySet
|
2019-11-07 11:10:46 -05:00
|
|
|
from .choices import *
|
2016-12-14 13:47:22 -05:00
|
|
|
|
|
|
|
|
2020-01-14 12:01:23 -05:00
|
|
|
__all__ = (
|
|
|
|
'Circuit',
|
|
|
|
'CircuitTermination',
|
|
|
|
'CircuitType',
|
2021-04-01 10:21:41 -04:00
|
|
|
'ProviderNetwork',
|
2020-01-14 12:01:23 -05:00
|
|
|
'Provider',
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2020-08-21 11:57:46 -04:00
|
|
|
@extras_features('custom_fields', 'custom_links', 'export_templates', 'webhooks')
|
2021-02-24 21:01:16 -05:00
|
|
|
class Provider(PrimaryModel):
|
2016-03-01 11:23:03 -05:00
|
|
|
"""
|
2016-06-21 12:45:02 -04:00
|
|
|
Each Circuit belongs to a Provider. This is usually a telecommunications company or similar organization. This model
|
|
|
|
stores information pertinent to the user's relationship with the Provider.
|
2016-03-01 11:23:03 -05:00
|
|
|
"""
|
2018-03-30 13:57:26 -04:00
|
|
|
name = models.CharField(
|
2020-10-15 15:37:34 -04:00
|
|
|
max_length=100,
|
2018-03-30 13:57:26 -04:00
|
|
|
unique=True
|
|
|
|
)
|
|
|
|
slug = models.SlugField(
|
2020-10-15 15:37:34 -04:00
|
|
|
max_length=100,
|
2018-03-30 13:57:26 -04:00
|
|
|
unique=True
|
|
|
|
)
|
|
|
|
asn = ASNField(
|
|
|
|
blank=True,
|
|
|
|
null=True,
|
2020-05-01 15:40:34 -04:00
|
|
|
verbose_name='ASN',
|
|
|
|
help_text='32-bit autonomous system number'
|
2018-03-30 13:57:26 -04:00
|
|
|
)
|
|
|
|
account = models.CharField(
|
|
|
|
max_length=30,
|
|
|
|
blank=True,
|
|
|
|
verbose_name='Account number'
|
|
|
|
)
|
|
|
|
portal_url = models.URLField(
|
|
|
|
blank=True,
|
2020-05-01 15:40:34 -04:00
|
|
|
verbose_name='Portal URL'
|
2018-03-30 13:57:26 -04:00
|
|
|
)
|
|
|
|
noc_contact = models.TextField(
|
|
|
|
blank=True,
|
|
|
|
verbose_name='NOC contact'
|
|
|
|
)
|
|
|
|
admin_contact = models.TextField(
|
|
|
|
blank=True,
|
|
|
|
verbose_name='Admin contact'
|
|
|
|
)
|
|
|
|
comments = models.TextField(
|
|
|
|
blank=True
|
|
|
|
)
|
2018-05-10 12:53:11 -04:00
|
|
|
|
2020-05-29 16:27:36 -04:00
|
|
|
objects = RestrictedQuerySet.as_manager()
|
|
|
|
|
2019-12-06 16:13:52 -05:00
|
|
|
csv_headers = [
|
|
|
|
'name', 'slug', 'asn', 'account', 'portal_url', 'noc_contact', 'admin_contact', 'comments',
|
|
|
|
]
|
|
|
|
clone_fields = [
|
|
|
|
'asn', 'account', 'portal_url', 'noc_contact', 'admin_contact',
|
|
|
|
]
|
2018-05-30 11:19:10 -04:00
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
class Meta:
|
|
|
|
ordering = ['name']
|
|
|
|
|
2017-01-23 22:44:29 +01:00
|
|
|
def __str__(self):
|
2016-03-01 11:23:03 -05:00
|
|
|
return self.name
|
|
|
|
|
|
|
|
def get_absolute_url(self):
|
2021-02-26 17:23:23 -05:00
|
|
|
return reverse('circuits:provider', args=[self.pk])
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2016-06-16 13:57:43 -04:00
|
|
|
def to_csv(self):
|
2018-02-02 11:34:31 -05:00
|
|
|
return (
|
2016-06-16 13:57:43 -04:00
|
|
|
self.name,
|
|
|
|
self.slug,
|
2017-01-04 10:47:00 -05:00
|
|
|
self.asn,
|
2016-06-16 13:57:43 -04:00
|
|
|
self.account,
|
|
|
|
self.portal_url,
|
2018-02-02 14:26:16 -05:00
|
|
|
self.noc_contact,
|
|
|
|
self.admin_contact,
|
|
|
|
self.comments,
|
2018-02-02 11:34:31 -05:00
|
|
|
)
|
2016-06-16 13:57:43 -04:00
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2021-03-18 11:10:48 -04:00
|
|
|
#
|
2021-04-01 10:21:41 -04:00
|
|
|
# Provider networks
|
2021-03-18 11:10:48 -04:00
|
|
|
#
|
|
|
|
|
|
|
|
@extras_features('custom_fields', 'custom_links', 'export_templates', 'webhooks')
|
2021-04-01 10:21:41 -04:00
|
|
|
class ProviderNetwork(PrimaryModel):
|
|
|
|
"""
|
|
|
|
This represents a provider network which exists outside of NetBox, the details of which are unknown or
|
|
|
|
unimportant to the user.
|
|
|
|
"""
|
2021-03-18 11:10:48 -04:00
|
|
|
name = models.CharField(
|
|
|
|
max_length=100
|
|
|
|
)
|
|
|
|
provider = models.ForeignKey(
|
|
|
|
to='circuits.Provider',
|
|
|
|
on_delete=models.PROTECT,
|
2021-04-01 10:21:41 -04:00
|
|
|
related_name='networks'
|
2021-03-18 11:10:48 -04:00
|
|
|
)
|
|
|
|
description = models.CharField(
|
|
|
|
max_length=200,
|
|
|
|
blank=True
|
|
|
|
)
|
|
|
|
comments = models.TextField(
|
|
|
|
blank=True
|
|
|
|
)
|
|
|
|
|
|
|
|
csv_headers = [
|
|
|
|
'provider', 'name', 'description', 'comments',
|
|
|
|
]
|
|
|
|
|
|
|
|
objects = RestrictedQuerySet.as_manager()
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
ordering = ('provider', 'name')
|
|
|
|
constraints = (
|
|
|
|
models.UniqueConstraint(
|
|
|
|
fields=('provider', 'name'),
|
2021-04-01 10:21:41 -04:00
|
|
|
name='circuits_providernetwork_provider_name'
|
2021-03-18 11:10:48 -04:00
|
|
|
),
|
|
|
|
)
|
|
|
|
unique_together = ('provider', 'name')
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return self.name
|
|
|
|
|
|
|
|
def get_absolute_url(self):
|
2021-04-01 10:21:41 -04:00
|
|
|
return reverse('circuits:providernetwork', args=[self.pk])
|
2021-03-18 11:10:48 -04:00
|
|
|
|
|
|
|
def to_csv(self):
|
|
|
|
return (
|
|
|
|
self.provider.name,
|
|
|
|
self.name,
|
|
|
|
self.description,
|
|
|
|
self.comments,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2021-05-06 13:01:20 -04:00
|
|
|
@extras_features('custom_fields', 'custom_links', 'export_templates', 'webhooks')
|
2021-02-24 21:01:16 -05:00
|
|
|
class CircuitType(OrganizationalModel):
|
2016-03-01 11:23:03 -05:00
|
|
|
"""
|
2017-01-04 10:47:00 -05:00
|
|
|
Circuits can be organized by their functional role. For example, a user might wish to define CircuitTypes named
|
2016-06-21 12:45:02 -04:00
|
|
|
"Long Haul," "Metro," or "Out-of-Band".
|
2016-03-01 11:23:03 -05:00
|
|
|
"""
|
2018-03-30 13:57:26 -04:00
|
|
|
name = models.CharField(
|
2020-10-15 15:37:34 -04:00
|
|
|
max_length=100,
|
2018-03-30 13:57:26 -04:00
|
|
|
unique=True
|
|
|
|
)
|
|
|
|
slug = models.SlugField(
|
2020-10-15 15:37:34 -04:00
|
|
|
max_length=100,
|
2018-03-30 13:57:26 -04:00
|
|
|
unique=True
|
|
|
|
)
|
2019-12-10 13:24:02 -05:00
|
|
|
description = models.CharField(
|
2020-03-13 15:49:58 -04:00
|
|
|
max_length=200,
|
2019-12-10 13:24:02 -05:00
|
|
|
blank=True,
|
|
|
|
)
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2020-05-29 16:27:36 -04:00
|
|
|
objects = RestrictedQuerySet.as_manager()
|
|
|
|
|
2019-12-10 13:24:02 -05:00
|
|
|
csv_headers = ['name', 'slug', 'description']
|
2018-02-02 14:26:16 -05:00
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
class Meta:
|
|
|
|
ordering = ['name']
|
|
|
|
|
2017-01-23 22:44:29 +01:00
|
|
|
def __str__(self):
|
2016-03-01 11:23:03 -05:00
|
|
|
return self.name
|
|
|
|
|
2016-05-13 12:44:03 -04:00
|
|
|
def get_absolute_url(self):
|
2021-03-26 14:44:43 -04:00
|
|
|
return reverse('circuits:circuittype', args=[self.pk])
|
2016-05-13 12:44:03 -04:00
|
|
|
|
2018-02-02 14:26:16 -05:00
|
|
|
def to_csv(self):
|
|
|
|
return (
|
|
|
|
self.name,
|
|
|
|
self.slug,
|
2019-12-10 13:24:02 -05:00
|
|
|
self.description,
|
2018-02-02 14:26:16 -05:00
|
|
|
)
|
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2020-03-14 03:03:22 -04:00
|
|
|
@extras_features('custom_fields', 'custom_links', 'export_templates', 'webhooks')
|
2021-02-24 21:01:16 -05:00
|
|
|
class Circuit(PrimaryModel):
|
2016-03-01 11:23:03 -05:00
|
|
|
"""
|
2016-06-21 12:45:02 -04:00
|
|
|
A communications circuit connects two points. Each Circuit belongs to a Provider; Providers may have multiple
|
2018-10-30 12:16:22 -04:00
|
|
|
circuits. Each circuit is also assigned a CircuitType and a Site. Circuit port speed and commit rate are measured
|
|
|
|
in Kbps.
|
2016-03-01 11:23:03 -05:00
|
|
|
"""
|
2018-03-30 13:57:26 -04:00
|
|
|
cid = models.CharField(
|
2020-10-15 15:37:34 -04:00
|
|
|
max_length=100,
|
2018-03-30 13:57:26 -04:00
|
|
|
verbose_name='Circuit ID'
|
|
|
|
)
|
|
|
|
provider = models.ForeignKey(
|
|
|
|
to='circuits.Provider',
|
|
|
|
on_delete=models.PROTECT,
|
|
|
|
related_name='circuits'
|
|
|
|
)
|
|
|
|
type = models.ForeignKey(
|
|
|
|
to='CircuitType',
|
|
|
|
on_delete=models.PROTECT,
|
|
|
|
related_name='circuits'
|
|
|
|
)
|
2019-11-07 11:10:46 -05:00
|
|
|
status = models.CharField(
|
|
|
|
max_length=50,
|
|
|
|
choices=CircuitStatusChoices,
|
|
|
|
default=CircuitStatusChoices.STATUS_ACTIVE
|
2018-03-30 13:57:26 -04:00
|
|
|
)
|
|
|
|
tenant = models.ForeignKey(
|
|
|
|
to='tenancy.Tenant',
|
|
|
|
on_delete=models.PROTECT,
|
|
|
|
related_name='circuits',
|
|
|
|
blank=True,
|
|
|
|
null=True
|
|
|
|
)
|
|
|
|
install_date = models.DateField(
|
|
|
|
blank=True,
|
|
|
|
null=True,
|
|
|
|
verbose_name='Date installed'
|
|
|
|
)
|
|
|
|
commit_rate = models.PositiveIntegerField(
|
|
|
|
blank=True,
|
|
|
|
null=True,
|
|
|
|
verbose_name='Commit rate (Kbps)')
|
|
|
|
description = models.CharField(
|
2020-03-13 15:49:58 -04:00
|
|
|
max_length=200,
|
2018-03-30 13:57:26 -04:00
|
|
|
blank=True
|
|
|
|
)
|
|
|
|
comments = models.TextField(
|
|
|
|
blank=True
|
|
|
|
)
|
2016-03-01 11:23:03 -05:00
|
|
|
|
2021-03-18 13:54:05 -04:00
|
|
|
# Cache associated CircuitTerminations
|
|
|
|
termination_a = models.ForeignKey(
|
|
|
|
to='circuits.CircuitTermination',
|
|
|
|
on_delete=models.SET_NULL,
|
|
|
|
related_name='+',
|
|
|
|
editable=False,
|
|
|
|
blank=True,
|
|
|
|
null=True
|
|
|
|
)
|
|
|
|
termination_z = models.ForeignKey(
|
|
|
|
to='circuits.CircuitTermination',
|
|
|
|
on_delete=models.SET_NULL,
|
|
|
|
related_name='+',
|
|
|
|
editable=False,
|
|
|
|
blank=True,
|
|
|
|
null=True
|
|
|
|
)
|
|
|
|
|
|
|
|
objects = RestrictedQuerySet.as_manager()
|
2018-05-10 12:53:11 -04:00
|
|
|
|
2018-02-06 14:58:11 -05:00
|
|
|
csv_headers = [
|
|
|
|
'cid', 'provider', 'type', 'status', 'tenant', 'install_date', 'commit_rate', 'description', 'comments',
|
|
|
|
]
|
2019-12-06 16:13:52 -05:00
|
|
|
clone_fields = [
|
|
|
|
'provider', 'type', 'status', 'tenant', 'install_date', 'commit_rate', 'description',
|
|
|
|
]
|
2017-06-09 16:24:59 -04:00
|
|
|
|
2016-03-01 11:23:03 -05:00
|
|
|
class Meta:
|
|
|
|
ordering = ['provider', 'cid']
|
|
|
|
unique_together = ['provider', 'cid']
|
|
|
|
|
2017-01-23 22:44:29 +01:00
|
|
|
def __str__(self):
|
2018-12-11 11:15:45 -05:00
|
|
|
return self.cid
|
2016-03-01 11:23:03 -05:00
|
|
|
|
|
|
|
def get_absolute_url(self):
|
|
|
|
return reverse('circuits:circuit', args=[self.pk])
|
2016-06-16 13:57:43 -04:00
|
|
|
|
|
|
|
def to_csv(self):
|
2018-02-02 11:34:31 -05:00
|
|
|
return (
|
2016-06-16 13:57:43 -04:00
|
|
|
self.cid,
|
|
|
|
self.provider.name,
|
|
|
|
self.type.name,
|
2018-02-06 14:06:05 -05:00
|
|
|
self.get_status_display(),
|
2017-01-04 10:47:00 -05:00
|
|
|
self.tenant.name if self.tenant else None,
|
2018-02-02 11:34:31 -05:00
|
|
|
self.install_date,
|
2017-01-04 10:47:00 -05:00
|
|
|
self.commit_rate,
|
2017-01-17 15:18:03 -05:00
|
|
|
self.description,
|
2018-02-02 14:26:16 -05:00
|
|
|
self.comments,
|
2018-02-02 11:34:31 -05:00
|
|
|
)
|
2016-06-20 15:58:18 -04:00
|
|
|
|
2018-02-06 14:06:05 -05:00
|
|
|
def get_status_class(self):
|
2020-09-24 16:35:53 -04:00
|
|
|
return CircuitStatusChoices.CSS_CLASSES.get(self.status)
|
2018-02-06 14:06:05 -05:00
|
|
|
|
2016-12-14 13:47:22 -05:00
|
|
|
|
2021-03-10 14:49:02 -05:00
|
|
|
@extras_features('webhooks')
|
2021-04-01 14:31:10 -04:00
|
|
|
class CircuitTermination(ChangeLoggedModel, CableTermination):
|
2018-03-30 13:57:26 -04:00
|
|
|
circuit = models.ForeignKey(
|
|
|
|
to='circuits.Circuit',
|
|
|
|
on_delete=models.CASCADE,
|
|
|
|
related_name='terminations'
|
|
|
|
)
|
|
|
|
term_side = models.CharField(
|
|
|
|
max_length=1,
|
2019-11-21 21:28:59 -05:00
|
|
|
choices=CircuitTerminationSideChoices,
|
2018-03-30 13:57:26 -04:00
|
|
|
verbose_name='Termination'
|
|
|
|
)
|
|
|
|
site = models.ForeignKey(
|
|
|
|
to='dcim.Site',
|
|
|
|
on_delete=models.PROTECT,
|
2021-03-18 11:57:59 -04:00
|
|
|
related_name='circuit_terminations',
|
|
|
|
blank=True,
|
|
|
|
null=True
|
|
|
|
)
|
2021-04-01 10:21:41 -04:00
|
|
|
provider_network = models.ForeignKey(
|
|
|
|
to=ProviderNetwork,
|
2021-03-18 11:57:59 -04:00
|
|
|
on_delete=models.PROTECT,
|
|
|
|
related_name='circuit_terminations',
|
|
|
|
blank=True,
|
|
|
|
null=True
|
2018-03-30 13:57:26 -04:00
|
|
|
)
|
|
|
|
port_speed = models.PositiveIntegerField(
|
2020-10-09 14:06:24 -04:00
|
|
|
verbose_name='Port speed (Kbps)',
|
|
|
|
blank=True,
|
|
|
|
null=True
|
2017-04-05 14:40:25 -04:00
|
|
|
)
|
2017-04-19 13:19:30 -04:00
|
|
|
upstream_speed = models.PositiveIntegerField(
|
2018-03-30 13:57:26 -04:00
|
|
|
blank=True,
|
|
|
|
null=True,
|
|
|
|
verbose_name='Upstream speed (Kbps)',
|
2017-04-19 13:19:30 -04:00
|
|
|
help_text='Upstream speed, if different from port speed'
|
|
|
|
)
|
2018-03-30 13:57:26 -04:00
|
|
|
xconnect_id = models.CharField(
|
|
|
|
max_length=50,
|
|
|
|
blank=True,
|
|
|
|
verbose_name='Cross-connect ID'
|
|
|
|
)
|
|
|
|
pp_info = models.CharField(
|
|
|
|
max_length=100,
|
|
|
|
blank=True,
|
|
|
|
verbose_name='Patch panel/port(s)'
|
|
|
|
)
|
2018-11-05 13:53:22 -05:00
|
|
|
description = models.CharField(
|
2020-03-13 15:49:58 -04:00
|
|
|
max_length=200,
|
2018-11-05 13:53:22 -05:00
|
|
|
blank=True
|
|
|
|
)
|
2016-12-14 13:47:22 -05:00
|
|
|
|
2020-05-29 16:27:36 -04:00
|
|
|
objects = RestrictedQuerySet.as_manager()
|
|
|
|
|
2016-12-14 13:47:22 -05:00
|
|
|
class Meta:
|
|
|
|
ordering = ['circuit', 'term_side']
|
|
|
|
unique_together = ['circuit', 'term_side']
|
|
|
|
|
2017-01-23 22:44:29 +01:00
|
|
|
def __str__(self):
|
2021-04-01 21:03:00 -04:00
|
|
|
return f'Termination {self.term_side}: {self.site or self.provider_network}'
|
2021-03-18 14:49:06 -04:00
|
|
|
|
|
|
|
def get_absolute_url(self):
|
|
|
|
if self.site:
|
|
|
|
return self.site.get_absolute_url()
|
2021-04-01 10:21:41 -04:00
|
|
|
return self.provider_network.get_absolute_url()
|
2021-03-18 11:57:59 -04:00
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
super().clean()
|
|
|
|
|
2021-04-01 10:21:41 -04:00
|
|
|
# Must define either site *or* provider network
|
|
|
|
if self.site is None and self.provider_network is None:
|
|
|
|
raise ValidationError("A circuit termination must attach to either a site or a provider network.")
|
|
|
|
if self.site and self.provider_network:
|
|
|
|
raise ValidationError("A circuit termination cannot attach to both a site and a provider network.")
|
2016-12-14 13:47:22 -05:00
|
|
|
|
2019-08-26 16:52:05 -04:00
|
|
|
def to_objectchange(self, action):
|
|
|
|
# Annotate the parent Circuit
|
2019-05-29 17:17:06 -04:00
|
|
|
try:
|
2021-03-04 13:06:04 -05:00
|
|
|
circuit = self.circuit
|
2019-05-29 17:17:06 -04:00
|
|
|
except Circuit.DoesNotExist:
|
|
|
|
# Parent circuit has been deleted
|
2021-03-04 13:06:04 -05:00
|
|
|
circuit = None
|
|
|
|
return super().to_objectchange(action, related_object=circuit)
|
2018-07-12 13:46:30 -04:00
|
|
|
|
2019-01-31 12:21:43 -05:00
|
|
|
@property
|
2021-03-05 13:06:21 -05:00
|
|
|
def parent_object(self):
|
2019-01-31 12:21:43 -05:00
|
|
|
return self.circuit
|
|
|
|
|
2016-12-14 13:47:22 -05:00
|
|
|
def get_peer_termination(self):
|
|
|
|
peer_side = 'Z' if self.term_side == 'A' else 'A'
|
|
|
|
try:
|
2020-07-23 12:48:03 -04:00
|
|
|
return CircuitTermination.objects.prefetch_related('site').get(
|
2020-07-07 14:13:58 -04:00
|
|
|
circuit=self.circuit,
|
|
|
|
term_side=peer_side
|
|
|
|
)
|
2016-12-14 13:47:22 -05:00
|
|
|
except CircuitTermination.DoesNotExist:
|
|
|
|
return None
|