mirror of
https://github.com/netbox-community/netbox.git
synced 2025-12-17 09:45:57 +00:00
* Convert ObjectType to a concrete child model of ContentType * Add public flag to ObjectType * Catch post_migrate signal to update ObjectTypes * Reference ObjectType records instead of registry for feature support * Automatically create ObjectTypes * Introduce has_feature() utility function * ObjectTypeManager should not inherit from ContentTypeManager * Misc cleanup * Don't populate ObjectTypes during migration * Don't automatically create ObjectTypes when a ContentType is created * Fix test * Extend has_feature() to accept a model or OT/CT * Misc cleanup * Deprecate get_for_id() on ObjectTypeManager * Rename contenttypes.py to object_types.py * Add index to features ArrayField * Keep FK & M2M fields pointing to ContentType * Add get_for_models() to ObjectTypeManager * Add tests for manager methods & utility functions * Fix migrations for M2M relations to ObjectType * model_is_public() should return False for non-core & non-plugin models * Order ObjectType by app_label & model name * Resolve migrations conflict
255 lines
7.4 KiB
Python
255 lines
7.4 KiB
Python
from collections import defaultdict
|
|
|
|
from django.conf import settings
|
|
from django.core.validators import ValidationError
|
|
from django.db import models
|
|
from django.urls import reverse
|
|
from django.utils.translation import gettext_lazy as _
|
|
|
|
from core.models import ObjectType
|
|
from extras.models.mixins import RenderTemplateMixin
|
|
from extras.querysets import ConfigContextQuerySet
|
|
from netbox.models import ChangeLoggedModel
|
|
from netbox.models.features import CloningMixin, CustomLinksMixin, ExportTemplatesMixin, SyncedDataMixin, TagsMixin
|
|
from utilities.data import deepmerge
|
|
|
|
__all__ = (
|
|
'ConfigContext',
|
|
'ConfigContextModel',
|
|
'ConfigTemplate',
|
|
)
|
|
|
|
|
|
#
|
|
# Config contexts
|
|
#
|
|
|
|
class ConfigContext(SyncedDataMixin, CloningMixin, CustomLinksMixin, ChangeLoggedModel):
|
|
"""
|
|
A ConfigContext represents a set of arbitrary data available to any Device or VirtualMachine matching its assigned
|
|
qualifiers (region, site, etc.). For example, the data stored in a ConfigContext assigned to site A and tenant B
|
|
will be available to a Device in site A assigned to tenant B. Data is stored in JSON format.
|
|
"""
|
|
name = models.CharField(
|
|
verbose_name=_('name'),
|
|
max_length=100,
|
|
unique=True
|
|
)
|
|
weight = models.PositiveSmallIntegerField(
|
|
verbose_name=_('weight'),
|
|
default=1000
|
|
)
|
|
description = models.CharField(
|
|
verbose_name=_('description'),
|
|
max_length=200,
|
|
blank=True
|
|
)
|
|
is_active = models.BooleanField(
|
|
verbose_name=_('is active'),
|
|
default=True,
|
|
)
|
|
regions = models.ManyToManyField(
|
|
to='dcim.Region',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
site_groups = models.ManyToManyField(
|
|
to='dcim.SiteGroup',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
sites = models.ManyToManyField(
|
|
to='dcim.Site',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
locations = models.ManyToManyField(
|
|
to='dcim.Location',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
device_types = models.ManyToManyField(
|
|
to='dcim.DeviceType',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
roles = models.ManyToManyField(
|
|
to='dcim.DeviceRole',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
platforms = models.ManyToManyField(
|
|
to='dcim.Platform',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
cluster_types = models.ManyToManyField(
|
|
to='virtualization.ClusterType',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
cluster_groups = models.ManyToManyField(
|
|
to='virtualization.ClusterGroup',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
clusters = models.ManyToManyField(
|
|
to='virtualization.Cluster',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
tenant_groups = models.ManyToManyField(
|
|
to='tenancy.TenantGroup',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
tenants = models.ManyToManyField(
|
|
to='tenancy.Tenant',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
tags = models.ManyToManyField(
|
|
to='extras.Tag',
|
|
related_name='+',
|
|
blank=True
|
|
)
|
|
data = models.JSONField()
|
|
|
|
objects = ConfigContextQuerySet.as_manager()
|
|
|
|
clone_fields = (
|
|
'weight', 'is_active', 'regions', 'site_groups', 'sites', 'locations', 'device_types',
|
|
'roles', 'platforms', 'cluster_types', 'cluster_groups', 'clusters', 'tenant_groups',
|
|
'tenants', 'tags', 'data',
|
|
)
|
|
|
|
class Meta:
|
|
ordering = ['weight', 'name']
|
|
verbose_name = _('config context')
|
|
verbose_name_plural = _('config contexts')
|
|
|
|
def __str__(self):
|
|
return self.name
|
|
|
|
def get_absolute_url(self):
|
|
return reverse('extras:configcontext', kwargs={'pk': self.pk})
|
|
|
|
@property
|
|
def docs_url(self):
|
|
return f'{settings.STATIC_URL}docs/models/extras/configcontext/'
|
|
|
|
def clean(self):
|
|
super().clean()
|
|
|
|
# Verify that JSON data is provided as an object
|
|
if type(self.data) is not dict:
|
|
raise ValidationError(
|
|
{'data': _('JSON data must be in object form. Example:') + ' {"foo": 123}'}
|
|
)
|
|
|
|
def sync_data(self):
|
|
"""
|
|
Synchronize context data from the designated DataFile (if any).
|
|
"""
|
|
self.data = self.data_file.get_data()
|
|
sync_data.alters_data = True
|
|
|
|
|
|
class ConfigContextModel(models.Model):
|
|
"""
|
|
A model which includes local configuration context data. This local data will override any inherited data from
|
|
ConfigContexts.
|
|
"""
|
|
local_context_data = models.JSONField(
|
|
blank=True,
|
|
null=True,
|
|
help_text=_(
|
|
"Local config context data takes precedence over source contexts in the final rendered config context"
|
|
)
|
|
)
|
|
|
|
class Meta:
|
|
abstract = True
|
|
|
|
def get_config_context(self):
|
|
"""
|
|
Compile all config data, overwriting lower-weight values with higher-weight values where a collision occurs.
|
|
Return the rendered configuration context for a device or VM.
|
|
"""
|
|
data = {}
|
|
|
|
if not hasattr(self, 'config_context_data'):
|
|
# The annotation is not available, so we fall back to manually querying for the config context objects
|
|
config_context_data = ConfigContext.objects.get_for_object(self, aggregate_data=True) or []
|
|
else:
|
|
# The attribute may exist, but the annotated value could be None if there is no config context data
|
|
config_context_data = self.config_context_data or []
|
|
|
|
for context in config_context_data:
|
|
data = deepmerge(data, context)
|
|
|
|
# If the object has local config context data defined, merge it last
|
|
if self.local_context_data:
|
|
data = deepmerge(data, self.local_context_data)
|
|
|
|
return data
|
|
|
|
def clean(self):
|
|
super().clean()
|
|
|
|
# Verify that JSON data is provided as an object
|
|
if self.local_context_data is not None and type(self.local_context_data) is not dict:
|
|
raise ValidationError(
|
|
{'local_context_data': _('JSON data must be in object form. Example:') + ' {"foo": 123}'}
|
|
)
|
|
|
|
|
|
#
|
|
# Config templates
|
|
#
|
|
|
|
class ConfigTemplate(
|
|
RenderTemplateMixin, SyncedDataMixin, CustomLinksMixin, ExportTemplatesMixin, TagsMixin, ChangeLoggedModel
|
|
):
|
|
name = models.CharField(
|
|
verbose_name=_('name'),
|
|
max_length=100
|
|
)
|
|
description = models.CharField(
|
|
verbose_name=_('description'),
|
|
max_length=200,
|
|
blank=True
|
|
)
|
|
|
|
class Meta:
|
|
ordering = ('name',)
|
|
verbose_name = _('config template')
|
|
verbose_name_plural = _('config templates')
|
|
|
|
def __str__(self):
|
|
return self.name
|
|
|
|
def get_absolute_url(self):
|
|
return reverse('extras:configtemplate', args=[self.pk])
|
|
|
|
def sync_data(self):
|
|
"""
|
|
Synchronize template content from the designated DataFile (if any).
|
|
"""
|
|
self.template_code = self.data_file.data_as_string
|
|
sync_data.alters_data = True
|
|
|
|
def get_context(self, context=None, queryset=None):
|
|
_context = defaultdict(dict)
|
|
|
|
# Populate all public models for reference within the template
|
|
for object_type in ObjectType.objects.public():
|
|
if model := object_type.model_class():
|
|
_context[object_type.app_label][model.__name__] = model
|
|
|
|
# Apply the provided context data, if any
|
|
if context is not None:
|
|
_context.update(context)
|
|
|
|
return _context
|