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

248 lines
6.6 KiB
Python
Raw Normal View History

2022-08-18 15:11:03 -07:00
from django.apps import apps
from django.contrib.contenttypes.fields import GenericRelation
from django.core.exceptions import ValidationError
2016-03-01 11:23:03 -05:00
from django.db import models
from django.urls import reverse
2016-03-01 11:23:03 -05:00
2021-11-04 15:58:17 -04:00
from circuits.choices import *
from dcim.models import CabledObjectModel
from netbox.models import (
ChangeLoggedModel, CustomFieldsMixin, CustomLinksMixin, OrganizationalModel, NetBoxModel, TagsMixin,
)
2022-01-19 15:16:10 -05:00
from netbox.models.features import WebhooksMixin
__all__ = (
'Circuit',
'CircuitTermination',
'CircuitType',
)
class CircuitType(OrganizationalModel):
2016-03-01 11:23:03 -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(
max_length=100,
2018-03-30 13:57:26 -04:00
unique=True
)
slug = models.SlugField(
max_length=100,
2018-03-30 13:57:26 -04:00
unique=True
)
description = models.CharField(
max_length=200,
blank=True,
)
2016-03-01 11:23:03 -05:00
class Meta:
ordering = ['name']
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):
return reverse('circuits:circuittype', args=[self.pk])
2016-05-13 12:44:03 -04:00
2016-03-01 11:23:03 -05:00
2022-01-26 20:57:14 -05:00
class Circuit(NetBoxModel):
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
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(
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='Installed'
)
termination_date = models.DateField(
blank=True,
null=True,
verbose_name='Terminates'
2018-03-30 13:57:26 -04:00
)
commit_rate = models.PositiveIntegerField(
blank=True,
null=True,
verbose_name='Commit rate (Kbps)')
description = models.CharField(
max_length=200,
2018-03-30 13:57:26 -04:00
blank=True
)
comments = models.TextField(
blank=True
)
2021-10-18 15:09:57 -04:00
# Generic relations
contacts = GenericRelation(
to='tenancy.ContactAssignment'
)
images = GenericRelation(
to='extras.ImageAttachment'
)
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
)
clone_fields = (
'provider', 'type', 'status', 'tenant', 'install_date', 'termination_date', 'commit_rate', 'description',
)
2016-03-01 11:23:03 -05:00
class Meta:
ordering = ['provider', 'cid']
unique_together = ['provider', 'cid']
def __str__(self):
return self.cid
2016-03-01 11:23:03 -05:00
2022-08-18 15:11:03 -07:00
@classmethod
def get_prerequisite_models(cls):
return [apps.get_model('circuits.Provider'), CircuitType]
2016-03-01 11:23:03 -05:00
def get_absolute_url(self):
return reverse('circuits:circuit', args=[self.pk])
def get_status_color(self):
return CircuitStatusChoices.colors.get(self.status)
class CircuitTermination(
CustomFieldsMixin,
CustomLinksMixin,
TagsMixin,
WebhooksMixin,
ChangeLoggedModel,
CabledObjectModel
):
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,
choices=CircuitTerminationSideChoices,
2018-03-30 13:57:26 -04:00
verbose_name='Termination'
)
site = models.ForeignKey(
to='dcim.Site',
on_delete=models.PROTECT,
related_name='circuit_terminations',
blank=True,
null=True
)
2021-04-01 10:21:41 -04:00
provider_network = models.ForeignKey(
2021-11-04 15:58:17 -04:00
to='circuits.ProviderNetwork',
on_delete=models.PROTECT,
related_name='circuit_terminations',
blank=True,
null=True
2018-03-30 13:57:26 -04:00
)
port_speed = models.PositiveIntegerField(
verbose_name='Port speed (Kbps)',
blank=True,
null=True
)
upstream_speed = models.PositiveIntegerField(
2018-03-30 13:57:26 -04:00
blank=True,
null=True,
verbose_name='Upstream speed (Kbps)',
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)'
)
description = models.CharField(
max_length=200,
blank=True
)
class Meta:
ordering = ['circuit', 'term_side']
unique_together = ['circuit', 'term_side']
def __str__(self):
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()
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.")
def to_objectchange(self, action):
2022-01-26 20:25:23 -05:00
objectchange = super().to_objectchange(action)
objectchange.related_object = self.circuit
return objectchange
@property
def parent_object(self):
return self.circuit
def get_peer_termination(self):
peer_side = 'Z' if self.term_side == 'A' else 'A'
try:
return CircuitTermination.objects.prefetch_related('site').get(
circuit=self.circuit,
term_side=peer_side
)
except CircuitTermination.DoesNotExist:
return None