2022-06-27 23:24:50 -05:00
|
|
|
from django.contrib.contenttypes.fields import GenericRelation, GenericForeignKey
|
|
|
|
from django.contrib.contenttypes.models import ContentType
|
|
|
|
from django.core.exceptions import ValidationError
|
|
|
|
from django.db import models
|
|
|
|
from django.urls import reverse
|
2022-08-23 16:17:40 -07:00
|
|
|
from django.utils.functional import cached_property
|
2023-07-31 22:28:07 +07:00
|
|
|
from django.utils.translation import gettext_lazy as _
|
2022-06-27 23:24:50 -05:00
|
|
|
|
|
|
|
from ipam.choices import L2VPNTypeChoices
|
|
|
|
from ipam.constants import L2VPN_ASSIGNMENT_MODELS
|
2022-11-04 08:28:09 -04:00
|
|
|
from netbox.models import NetBoxModel, PrimaryModel
|
2022-06-27 23:24:50 -05:00
|
|
|
|
2022-08-17 10:51:50 -04:00
|
|
|
__all__ = (
|
|
|
|
'L2VPN',
|
|
|
|
'L2VPNTermination',
|
|
|
|
)
|
|
|
|
|
2022-06-27 23:24:50 -05:00
|
|
|
|
2022-11-04 08:28:09 -04:00
|
|
|
class L2VPN(PrimaryModel):
|
2022-06-27 23:24:50 -05:00
|
|
|
name = models.CharField(
|
2023-07-31 22:28:07 +07:00
|
|
|
verbose_name=_('name'),
|
2022-06-27 23:24:50 -05:00
|
|
|
max_length=100,
|
|
|
|
unique=True
|
|
|
|
)
|
2022-08-24 14:12:14 -07:00
|
|
|
slug = models.SlugField(
|
2023-07-31 22:28:07 +07:00
|
|
|
verbose_name=_('slug'),
|
2022-08-24 14:12:14 -07:00
|
|
|
max_length=100,
|
|
|
|
unique=True
|
|
|
|
)
|
2022-08-17 10:47:42 -04:00
|
|
|
type = models.CharField(
|
2023-07-31 22:28:07 +07:00
|
|
|
verbose_name=_('type'),
|
2022-08-17 10:47:42 -04:00
|
|
|
max_length=50,
|
|
|
|
choices=L2VPNTypeChoices
|
|
|
|
)
|
2022-06-27 23:24:50 -05:00
|
|
|
identifier = models.BigIntegerField(
|
2023-07-31 22:28:07 +07:00
|
|
|
verbose_name=_('identifier'),
|
2022-06-27 23:24:50 -05:00
|
|
|
null=True,
|
2022-08-17 10:47:42 -04:00
|
|
|
blank=True
|
2022-06-27 23:24:50 -05:00
|
|
|
)
|
|
|
|
import_targets = models.ManyToManyField(
|
|
|
|
to='ipam.RouteTarget',
|
|
|
|
related_name='importing_l2vpns',
|
|
|
|
blank=True,
|
|
|
|
)
|
|
|
|
export_targets = models.ManyToManyField(
|
|
|
|
to='ipam.RouteTarget',
|
|
|
|
related_name='exporting_l2vpns',
|
|
|
|
blank=True
|
|
|
|
)
|
|
|
|
tenant = models.ForeignKey(
|
|
|
|
to='tenancy.Tenant',
|
2022-07-06 08:17:50 -05:00
|
|
|
on_delete=models.PROTECT,
|
2022-06-27 23:24:50 -05:00
|
|
|
related_name='l2vpns',
|
|
|
|
blank=True,
|
|
|
|
null=True
|
|
|
|
)
|
|
|
|
contacts = GenericRelation(
|
|
|
|
to='tenancy.ContactAssignment'
|
|
|
|
)
|
|
|
|
|
2022-08-18 15:19:34 -04:00
|
|
|
clone_fields = ('type',)
|
|
|
|
|
2022-06-27 23:24:50 -05:00
|
|
|
class Meta:
|
2022-07-11 21:51:39 -04:00
|
|
|
ordering = ('name', 'identifier')
|
2022-06-27 23:24:50 -05:00
|
|
|
verbose_name = 'L2VPN'
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
if self.identifier:
|
|
|
|
return f'{self.name} ({self.identifier})'
|
|
|
|
return f'{self.name}'
|
|
|
|
|
|
|
|
def get_absolute_url(self):
|
|
|
|
return reverse('ipam:l2vpn', args=[self.pk])
|
|
|
|
|
2022-08-23 16:17:40 -07:00
|
|
|
@cached_property
|
2022-08-23 10:34:06 -07:00
|
|
|
def can_add_termination(self):
|
|
|
|
if self.type in L2VPNTypeChoices.P2P and self.terminations.count() >= 2:
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
|
2022-06-27 23:24:50 -05:00
|
|
|
|
|
|
|
class L2VPNTermination(NetBoxModel):
|
|
|
|
l2vpn = models.ForeignKey(
|
|
|
|
to='ipam.L2VPN',
|
|
|
|
on_delete=models.CASCADE,
|
2022-07-06 08:57:15 -05:00
|
|
|
related_name='terminations'
|
2022-06-27 23:24:50 -05:00
|
|
|
)
|
|
|
|
assigned_object_type = models.ForeignKey(
|
|
|
|
to=ContentType,
|
|
|
|
limit_choices_to=L2VPN_ASSIGNMENT_MODELS,
|
|
|
|
on_delete=models.PROTECT,
|
2022-07-06 08:57:15 -05:00
|
|
|
related_name='+'
|
2022-06-27 23:24:50 -05:00
|
|
|
)
|
2022-07-06 08:57:15 -05:00
|
|
|
assigned_object_id = models.PositiveBigIntegerField()
|
2022-06-27 23:24:50 -05:00
|
|
|
assigned_object = GenericForeignKey(
|
|
|
|
ct_field='assigned_object_type',
|
|
|
|
fk_field='assigned_object_id'
|
|
|
|
)
|
|
|
|
|
2022-08-17 10:51:50 -04:00
|
|
|
clone_fields = ('l2vpn',)
|
2022-11-16 17:22:09 -05:00
|
|
|
prerequisite_models = (
|
|
|
|
'ipam.L2VPN',
|
|
|
|
)
|
2022-08-17 10:51:50 -04:00
|
|
|
|
2022-06-27 23:24:50 -05:00
|
|
|
class Meta:
|
|
|
|
ordering = ('l2vpn',)
|
2022-07-11 21:51:39 -04:00
|
|
|
verbose_name = 'L2VPN termination'
|
2022-06-27 23:24:50 -05:00
|
|
|
constraints = (
|
|
|
|
models.UniqueConstraint(
|
|
|
|
fields=('assigned_object_type', 'assigned_object_id'),
|
|
|
|
name='ipam_l2vpntermination_assigned_object'
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
if self.pk is not None:
|
|
|
|
return f'{self.assigned_object} <> {self.l2vpn}'
|
2022-07-06 08:57:15 -05:00
|
|
|
return super().__str__()
|
2022-06-27 23:24:50 -05:00
|
|
|
|
|
|
|
def get_absolute_url(self):
|
|
|
|
return reverse('ipam:l2vpntermination', args=[self.pk])
|
|
|
|
|
|
|
|
def clean(self):
|
2022-07-06 08:57:15 -05:00
|
|
|
# Only check is assigned_object is set. Required otherwise we have an Integrity Error thrown.
|
2022-06-27 23:24:50 -05:00
|
|
|
if self.assigned_object:
|
|
|
|
obj_id = self.assigned_object.pk
|
|
|
|
obj_type = ContentType.objects.get_for_model(self.assigned_object)
|
|
|
|
if L2VPNTermination.objects.filter(assigned_object_id=obj_id, assigned_object_type=obj_type).\
|
|
|
|
exclude(pk=self.pk).count() > 0:
|
2023-07-31 22:28:07 +07:00
|
|
|
raise ValidationError(
|
|
|
|
_('L2VPN Termination already assigned ({assigned_object})').format(
|
|
|
|
assigned_object=self.assigned_object
|
|
|
|
)
|
|
|
|
)
|
2022-06-27 23:24:50 -05:00
|
|
|
|
|
|
|
# Only check if L2VPN is set and is of type P2P
|
2022-07-11 21:51:39 -04:00
|
|
|
if hasattr(self, 'l2vpn') and self.l2vpn.type in L2VPNTypeChoices.P2P:
|
2022-07-06 08:17:50 -05:00
|
|
|
terminations_count = L2VPNTermination.objects.filter(l2vpn=self.l2vpn).exclude(pk=self.pk).count()
|
|
|
|
if terminations_count >= 2:
|
|
|
|
l2vpn_type = self.l2vpn.get_type_display()
|
|
|
|
raise ValidationError(
|
2023-07-31 22:28:07 +07:00
|
|
|
_(
|
|
|
|
'{l2vpn_type} L2VPNs cannot have more than two terminations; found {terminations_count} '
|
|
|
|
'already defined.'
|
|
|
|
).format(l2vpn_type=l2vpn_type, terminations_count=terminations_count))
|
2022-08-02 12:38:16 -05:00
|
|
|
|
|
|
|
@property
|
|
|
|
def assigned_object_parent(self):
|
|
|
|
obj_type = ContentType.objects.get_for_model(self.assigned_object)
|
|
|
|
if obj_type.model == 'vminterface':
|
|
|
|
return self.assigned_object.virtual_machine
|
|
|
|
elif obj_type.model == 'interface':
|
|
|
|
return self.assigned_object.device
|
|
|
|
elif obj_type.model == 'vminterface':
|
|
|
|
return self.assigned_object.virtual_machine
|
|
|
|
return None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def assigned_object_site(self):
|
|
|
|
return self.assigned_object_parent.site
|