2021-10-12 17:02:53 -04:00
|
|
|
from django.core.exceptions import ValidationError
|
2021-10-12 12:27:12 -04:00
|
|
|
from django.db import models
|
2021-10-12 13:48:06 -04:00
|
|
|
from django.urls import reverse
|
2021-10-12 12:27:12 -04:00
|
|
|
|
2021-10-12 17:02:53 -04:00
|
|
|
from dcim.constants import WIRELESS_IFACE_TYPES
|
2021-10-12 12:27:12 -04:00
|
|
|
from extras.utils import extras_features
|
2021-10-12 13:48:06 -04:00
|
|
|
from netbox.models import BigIDModel, PrimaryModel
|
2021-10-12 12:27:12 -04:00
|
|
|
from utilities.querysets import RestrictedQuerySet
|
2021-10-13 09:46:17 -04:00
|
|
|
from .constants import SSID_MAX_LENGTH
|
2021-10-12 12:27:12 -04:00
|
|
|
|
|
|
|
__all__ = (
|
2021-10-12 17:02:53 -04:00
|
|
|
'WirelessLAN',
|
2021-10-13 09:46:17 -04:00
|
|
|
'WirelessLink',
|
2021-10-12 12:27:12 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@extras_features('custom_fields', 'custom_links', 'export_templates', 'tags', 'webhooks')
|
2021-10-12 17:02:53 -04:00
|
|
|
class WirelessLAN(PrimaryModel):
|
2021-10-12 12:27:12 -04:00
|
|
|
"""
|
2021-10-13 09:46:17 -04:00
|
|
|
A wireless network formed among an arbitrary number of access point and clients.
|
2021-10-12 12:27:12 -04:00
|
|
|
"""
|
2021-10-12 17:02:53 -04:00
|
|
|
ssid = models.CharField(
|
2021-10-13 09:46:17 -04:00
|
|
|
max_length=SSID_MAX_LENGTH,
|
|
|
|
verbose_name='SSID'
|
2021-10-12 12:27:12 -04:00
|
|
|
)
|
|
|
|
vlan = models.ForeignKey(
|
|
|
|
to='ipam.VLAN',
|
|
|
|
on_delete=models.PROTECT,
|
|
|
|
blank=True,
|
|
|
|
null=True,
|
|
|
|
verbose_name='VLAN'
|
|
|
|
)
|
|
|
|
description = models.CharField(
|
|
|
|
max_length=200,
|
|
|
|
blank=True
|
|
|
|
)
|
|
|
|
|
|
|
|
objects = RestrictedQuerySet.as_manager()
|
|
|
|
|
|
|
|
class Meta:
|
2021-10-12 17:02:53 -04:00
|
|
|
ordering = ('ssid', 'pk')
|
|
|
|
verbose_name = 'Wireless LAN'
|
2021-10-12 12:27:12 -04:00
|
|
|
|
|
|
|
def __str__(self):
|
2021-10-12 17:02:53 -04:00
|
|
|
return self.ssid
|
2021-10-12 13:48:06 -04:00
|
|
|
|
|
|
|
def get_absolute_url(self):
|
2021-10-13 09:46:17 -04:00
|
|
|
return reverse('wireless:wirelesslan', args=[self.pk])
|
|
|
|
|
|
|
|
|
|
|
|
@extras_features('custom_fields', 'custom_links', 'export_templates', 'tags', 'webhooks')
|
|
|
|
class WirelessLink(PrimaryModel):
|
|
|
|
"""
|
|
|
|
A point-to-point connection between two wireless Interfaces.
|
|
|
|
"""
|
|
|
|
interface_a = models.ForeignKey(
|
|
|
|
to='dcim.Interface',
|
|
|
|
limit_choices_to={'type__in': WIRELESS_IFACE_TYPES},
|
|
|
|
on_delete=models.PROTECT,
|
|
|
|
related_name='+'
|
|
|
|
)
|
|
|
|
interface_b = models.ForeignKey(
|
|
|
|
to='dcim.Interface',
|
|
|
|
limit_choices_to={'type__in': WIRELESS_IFACE_TYPES},
|
|
|
|
on_delete=models.PROTECT,
|
|
|
|
related_name='+'
|
|
|
|
)
|
|
|
|
ssid = models.CharField(
|
|
|
|
max_length=SSID_MAX_LENGTH,
|
|
|
|
blank=True,
|
|
|
|
verbose_name='SSID'
|
|
|
|
)
|
|
|
|
description = models.CharField(
|
|
|
|
max_length=200,
|
|
|
|
blank=True
|
|
|
|
)
|
|
|
|
|
|
|
|
# Cache the associated device for the A and B interfaces. This enables filtering of WirelessLinks by their
|
|
|
|
# associated Devices.
|
|
|
|
_interface_a_device = models.ForeignKey(
|
|
|
|
to='dcim.Device',
|
|
|
|
on_delete=models.CASCADE,
|
|
|
|
related_name='+',
|
|
|
|
blank=True,
|
|
|
|
null=True
|
|
|
|
)
|
|
|
|
_interface_b_device = models.ForeignKey(
|
|
|
|
to='dcim.Device',
|
|
|
|
on_delete=models.CASCADE,
|
|
|
|
related_name='+',
|
|
|
|
blank=True,
|
|
|
|
null=True
|
|
|
|
)
|
|
|
|
|
|
|
|
objects = RestrictedQuerySet.as_manager()
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
ordering = ['pk']
|
|
|
|
unique_together = ('interface_a', 'interface_b')
|
|
|
|
|
2021-10-13 14:04:53 -04:00
|
|
|
def __str__(self):
|
|
|
|
return f'#{self.pk}'
|
|
|
|
|
2021-10-13 09:46:17 -04:00
|
|
|
def get_absolute_url(self):
|
|
|
|
return reverse('wireless:wirelesslink', args=[self.pk])
|
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
|
|
|
|
# Validate interface types
|
|
|
|
if self.interface_a.type not in WIRELESS_IFACE_TYPES:
|
|
|
|
raise ValidationError({
|
|
|
|
'interface_a': f"{self.interface_a.get_type_display()} is not a wireless interface."
|
|
|
|
})
|
|
|
|
if self.interface_b.type not in WIRELESS_IFACE_TYPES:
|
|
|
|
raise ValidationError({
|
|
|
|
'interface_a': f"{self.interface_b.get_type_display()} is not a wireless interface."
|
|
|
|
})
|
|
|
|
|
|
|
|
def save(self, *args, **kwargs):
|
|
|
|
|
|
|
|
# Store the parent Device for the A and B interfaces
|
|
|
|
self._interface_a_device = self.interface_a.device
|
|
|
|
self._interface_b_device = self.interface_b.device
|
|
|
|
|
|
|
|
super().save(*args, **kwargs)
|