| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425 |
- from django.contrib.contenttypes.fields import GenericRelation
- from django.core.exceptions import ValidationError
- from django.db import models
- from django.urls import reverse
- from mptt.models import TreeForeignKey
- from timezone_field import TimeZoneField
- from dcim.choices import *
- from dcim.constants import *
- from netbox.models import NestedGroupModel, NetBoxModel
- from utilities.fields import NaturalOrderingField
- __all__ = (
- 'Location',
- 'Region',
- 'Site',
- 'SiteGroup',
- )
- #
- # Regions
- #
- class Region(NestedGroupModel):
- """
- A region represents a geographic collection of sites. For example, you might create regions representing countries,
- states, and/or cities. Regions are recursively nested into a hierarchy: all sites belonging to a child region are
- also considered to be members of its parent and ancestor region(s).
- """
- parent = TreeForeignKey(
- to='self',
- on_delete=models.CASCADE,
- related_name='children',
- blank=True,
- null=True,
- db_index=True
- )
- name = models.CharField(
- max_length=100
- )
- slug = models.SlugField(
- max_length=100
- )
- description = models.CharField(
- max_length=200,
- blank=True
- )
- # Generic relations
- vlan_groups = GenericRelation(
- to='ipam.VLANGroup',
- content_type_field='scope_type',
- object_id_field='scope_id',
- related_query_name='region'
- )
- contacts = GenericRelation(
- to='tenancy.ContactAssignment'
- )
- class Meta:
- constraints = (
- models.UniqueConstraint(
- fields=('parent', 'name'),
- name='dcim_region_parent_name'
- ),
- models.UniqueConstraint(
- fields=('name',),
- name='dcim_region_name',
- condition=Q(parent=None)
- ),
- models.UniqueConstraint(
- fields=('parent', 'slug'),
- name='dcim_region_parent_slug'
- ),
- models.UniqueConstraint(
- fields=('slug',),
- name='dcim_region_slug',
- condition=Q(parent=None)
- ),
- )
- def validate_unique(self, exclude=None):
- if self.parent is None:
- regions = Region.objects.exclude(pk=self.pk)
- if regions.filter(name=self.name, parent__isnull=True).exists():
- raise ValidationError({
- 'name': 'A region with this name already exists.'
- })
- if regions.filter(slug=self.slug, parent__isnull=True).exists():
- raise ValidationError({
- 'name': 'A region with this slug already exists.'
- })
- super().validate_unique(exclude=exclude)
- def get_absolute_url(self):
- return reverse('dcim:region', args=[self.pk])
- def get_site_count(self):
- return Site.objects.filter(
- Q(region=self) |
- Q(region__in=self.get_descendants())
- ).count()
- #
- # Site groups
- #
- class SiteGroup(NestedGroupModel):
- """
- A site group is an arbitrary grouping of sites. For example, you might have corporate sites and customer sites; and
- within corporate sites you might distinguish between offices and data centers. Like regions, site groups can be
- nested recursively to form a hierarchy.
- """
- parent = TreeForeignKey(
- to='self',
- on_delete=models.CASCADE,
- related_name='children',
- blank=True,
- null=True,
- db_index=True
- )
- name = models.CharField(
- max_length=100
- )
- slug = models.SlugField(
- max_length=100
- )
- description = models.CharField(
- max_length=200,
- blank=True
- )
- # Generic relations
- vlan_groups = GenericRelation(
- to='ipam.VLANGroup',
- content_type_field='scope_type',
- object_id_field='scope_id',
- related_query_name='site_group'
- )
- contacts = GenericRelation(
- to='tenancy.ContactAssignment'
- )
- class Meta:
- constraints = (
- models.UniqueConstraint(
- fields=('parent', 'name'),
- name='dcim_sitegroup_parent_name'
- ),
- models.UniqueConstraint(
- fields=('name',),
- name='dcim_sitegroup_name',
- condition=Q(parent=None)
- ),
- models.UniqueConstraint(
- fields=('parent', 'slug'),
- name='dcim_sitegroup_parent_slug'
- ),
- models.UniqueConstraint(
- fields=('slug',),
- name='dcim_sitegroup_slug',
- condition=Q(parent=None)
- ),
- )
- def validate_unique(self, exclude=None):
- if self.parent is None:
- site_groups = SiteGroup.objects.exclude(pk=self.pk)
- if site_groups.filter(name=self.name, parent__isnull=True).exists():
- raise ValidationError({
- 'name': 'A site group with this name already exists.'
- })
- if site_groups.filter(slug=self.slug, parent__isnull=True).exists():
- raise ValidationError({
- 'name': 'A site group with this slug already exists.'
- })
- super().validate_unique(exclude=exclude)
- def get_absolute_url(self):
- return reverse('dcim:sitegroup', args=[self.pk])
- def get_site_count(self):
- return Site.objects.filter(
- Q(group=self) |
- Q(group__in=self.get_descendants())
- ).count()
- #
- # Sites
- #
- class Site(NetBoxModel):
- """
- A Site represents a geographic location within a network; typically a building or campus. The optional facility
- field can be used to include an external designation, such as a data center name (e.g. Equinix SV6).
- """
- name = models.CharField(
- max_length=100,
- unique=True
- )
- _name = NaturalOrderingField(
- target_field='name',
- max_length=100,
- blank=True
- )
- slug = models.SlugField(
- max_length=100,
- unique=True
- )
- status = models.CharField(
- max_length=50,
- choices=SiteStatusChoices,
- default=SiteStatusChoices.STATUS_ACTIVE
- )
- region = models.ForeignKey(
- to='dcim.Region',
- on_delete=models.SET_NULL,
- related_name='sites',
- blank=True,
- null=True
- )
- group = models.ForeignKey(
- to='dcim.SiteGroup',
- on_delete=models.SET_NULL,
- related_name='sites',
- blank=True,
- null=True
- )
- tenant = models.ForeignKey(
- to='tenancy.Tenant',
- on_delete=models.PROTECT,
- related_name='sites',
- blank=True,
- null=True
- )
- facility = models.CharField(
- max_length=50,
- blank=True,
- help_text='Local facility ID or description'
- )
- asns = models.ManyToManyField(
- to='ipam.ASN',
- related_name='sites',
- blank=True
- )
- time_zone = TimeZoneField(
- blank=True
- )
- description = models.CharField(
- max_length=200,
- blank=True
- )
- physical_address = models.CharField(
- max_length=200,
- blank=True
- )
- shipping_address = models.CharField(
- max_length=200,
- blank=True
- )
- latitude = models.DecimalField(
- max_digits=8,
- decimal_places=6,
- blank=True,
- null=True,
- help_text='GPS coordinate (latitude)'
- )
- longitude = models.DecimalField(
- max_digits=9,
- decimal_places=6,
- blank=True,
- null=True,
- help_text='GPS coordinate (longitude)'
- )
- comments = models.TextField(
- blank=True
- )
- # Generic relations
- vlan_groups = GenericRelation(
- to='ipam.VLANGroup',
- content_type_field='scope_type',
- object_id_field='scope_id',
- related_query_name='site'
- )
- contacts = GenericRelation(
- to='tenancy.ContactAssignment'
- )
- images = GenericRelation(
- to='extras.ImageAttachment'
- )
- clone_fields = (
- 'status', 'region', 'group', 'tenant', 'facility', 'time_zone', 'physical_address', 'shipping_address',
- 'latitude', 'longitude', 'description',
- )
- class Meta:
- ordering = ('_name',)
- def __str__(self):
- return self.name
- def get_absolute_url(self):
- return reverse('dcim:site', args=[self.pk])
- def get_status_color(self):
- return SiteStatusChoices.colors.get(self.status)
- #
- # Locations
- #
- class Location(NestedGroupModel):
- """
- A Location represents a subgroup of Racks and/or Devices within a Site. A Location may represent a building within a
- site, or a room within a building, for example.
- """
- name = models.CharField(
- max_length=100
- )
- slug = models.SlugField(
- max_length=100
- )
- site = models.ForeignKey(
- to='dcim.Site',
- on_delete=models.CASCADE,
- related_name='locations'
- )
- parent = TreeForeignKey(
- to='self',
- on_delete=models.CASCADE,
- related_name='children',
- blank=True,
- null=True,
- db_index=True
- )
- status = models.CharField(
- max_length=50,
- choices=LocationStatusChoices,
- default=LocationStatusChoices.STATUS_ACTIVE
- )
- tenant = models.ForeignKey(
- to='tenancy.Tenant',
- on_delete=models.PROTECT,
- related_name='locations',
- blank=True,
- null=True
- )
- description = models.CharField(
- max_length=200,
- blank=True
- )
- # Generic relations
- vlan_groups = GenericRelation(
- to='ipam.VLANGroup',
- content_type_field='scope_type',
- object_id_field='scope_id',
- related_query_name='location'
- )
- contacts = GenericRelation(
- to='tenancy.ContactAssignment'
- )
- images = GenericRelation(
- to='extras.ImageAttachment'
- )
- clone_fields = ('site', 'parent', 'status', 'tenant', 'description')
- class Meta:
- ordering = ['site', 'name']
- constraints = (
- models.UniqueConstraint(
- fields=('site', 'parent', 'name'),
- name='dcim_location_parent_name'
- ),
- models.UniqueConstraint(
- fields=('site', 'name'),
- name='dcim_location_name',
- condition=Q(parent=None)
- ),
- models.UniqueConstraint(
- fields=('site', 'parent', 'slug'),
- name='dcim_location_parent_slug'
- ),
- models.UniqueConstraint(
- fields=('site', 'slug'),
- name='dcim_location_slug',
- condition=Q(parent=None)
- ),
- )
- def validate_unique(self, exclude=None):
- if self.parent is None:
- locations = Location.objects.exclude(pk=self.pk)
- if locations.filter(name=self.name, site=self.site, parent__isnull=True).exists():
- raise ValidationError({
- "name": f"A location with this name in site {self.site} already exists."
- })
- if locations.filter(slug=self.slug, site=self.site, parent__isnull=True).exists():
- raise ValidationError({
- "name": f"A location with this slug in site {self.site} already exists."
- })
- super().validate_unique(exclude=exclude)
- def get_absolute_url(self):
- return reverse('dcim:location', args=[self.pk])
- def get_status_color(self):
- return LocationStatusChoices.colors.get(self.status)
- def clean(self):
- super().clean()
- # Parent Location (if any) must belong to the same Site
- if self.parent and self.parent.site != self.site:
- raise ValidationError(f"Parent location ({self.parent}) must belong to the same site ({self.site})")
|