mirror of
https://github.com/netbox-community/netbox.git
synced 2024-05-10 07:54:54 +00:00
* WIP * Add bulk sync view for config contexts * Introduce 'sync' permission for synced data models * Docs & cleanup * Remove unused method * Add a REST API endpoint to synchronize config context data
294 lines
9.5 KiB
Python
294 lines
9.5 KiB
Python
from django import forms
|
|
from django.contrib.contenttypes.models import ContentType
|
|
from django.http import QueryDict
|
|
from django.utils.translation import gettext as _
|
|
|
|
from dcim.models import DeviceRole, DeviceType, Location, Platform, Region, Site, SiteGroup
|
|
from extras.choices import *
|
|
from extras.forms.mixins import SyncedDataMixin
|
|
from extras.models import *
|
|
from extras.utils import FeatureQuery
|
|
from netbox.forms import NetBoxModelForm
|
|
from tenancy.models import Tenant, TenantGroup
|
|
from utilities.forms import (
|
|
add_blank_choice, BootstrapMixin, CommentField, ContentTypeChoiceField, ContentTypeMultipleChoiceField,
|
|
DynamicModelMultipleChoiceField, JSONField, SlugField, StaticSelect,
|
|
)
|
|
from virtualization.models import Cluster, ClusterGroup, ClusterType
|
|
|
|
__all__ = (
|
|
'ConfigContextForm',
|
|
'CustomFieldForm',
|
|
'CustomLinkForm',
|
|
'ExportTemplateForm',
|
|
'ImageAttachmentForm',
|
|
'JournalEntryForm',
|
|
'SavedFilterForm',
|
|
'TagForm',
|
|
'WebhookForm',
|
|
)
|
|
|
|
|
|
class CustomFieldForm(BootstrapMixin, forms.ModelForm):
|
|
content_types = ContentTypeMultipleChoiceField(
|
|
queryset=ContentType.objects.all(),
|
|
limit_choices_to=FeatureQuery('custom_fields'),
|
|
)
|
|
object_type = ContentTypeChoiceField(
|
|
queryset=ContentType.objects.all(),
|
|
# TODO: Come up with a canonical way to register suitable models
|
|
limit_choices_to=FeatureQuery('webhooks'),
|
|
required=False,
|
|
help_text=_("Type of the related object (for object/multi-object fields only)")
|
|
)
|
|
|
|
fieldsets = (
|
|
('Custom Field', (
|
|
'content_types', 'name', 'label', 'group_name', 'type', 'object_type', 'required', 'description',
|
|
)),
|
|
('Behavior', ('search_weight', 'filter_logic', 'ui_visibility', 'weight')),
|
|
('Values', ('default', 'choices')),
|
|
('Validation', ('validation_minimum', 'validation_maximum', 'validation_regex')),
|
|
)
|
|
|
|
class Meta:
|
|
model = CustomField
|
|
fields = '__all__'
|
|
help_texts = {
|
|
'type': _("The type of data stored in this field. For object/multi-object fields, select the related object "
|
|
"type below.")
|
|
}
|
|
widgets = {
|
|
'type': StaticSelect(),
|
|
'filter_logic': StaticSelect(),
|
|
'ui_visibility': StaticSelect(),
|
|
}
|
|
|
|
|
|
class CustomLinkForm(BootstrapMixin, forms.ModelForm):
|
|
content_types = ContentTypeMultipleChoiceField(
|
|
queryset=ContentType.objects.all(),
|
|
limit_choices_to=FeatureQuery('custom_links')
|
|
)
|
|
|
|
fieldsets = (
|
|
('Custom Link', ('name', 'content_types', 'weight', 'group_name', 'button_class', 'enabled', 'new_window')),
|
|
('Templates', ('link_text', 'link_url')),
|
|
)
|
|
|
|
class Meta:
|
|
model = CustomLink
|
|
fields = '__all__'
|
|
widgets = {
|
|
'button_class': StaticSelect(),
|
|
'link_text': forms.Textarea(attrs={'class': 'font-monospace'}),
|
|
'link_url': forms.Textarea(attrs={'class': 'font-monospace'}),
|
|
}
|
|
help_texts = {
|
|
'link_text': _('Jinja2 template code for the link text. Reference the object as <code>{{ object }}</code>. '
|
|
'Links which render as empty text will not be displayed.'),
|
|
'link_url': _('Jinja2 template code for the link URL. Reference the object as <code>{{ object }}</code>.'),
|
|
}
|
|
|
|
|
|
class ExportTemplateForm(BootstrapMixin, forms.ModelForm):
|
|
content_types = ContentTypeMultipleChoiceField(
|
|
queryset=ContentType.objects.all(),
|
|
limit_choices_to=FeatureQuery('export_templates')
|
|
)
|
|
|
|
fieldsets = (
|
|
('Export Template', ('name', 'content_types', 'description')),
|
|
('Template', ('template_code',)),
|
|
('Rendering', ('mime_type', 'file_extension', 'as_attachment')),
|
|
)
|
|
|
|
class Meta:
|
|
model = ExportTemplate
|
|
fields = '__all__'
|
|
widgets = {
|
|
'template_code': forms.Textarea(attrs={'class': 'font-monospace'}),
|
|
}
|
|
|
|
|
|
class SavedFilterForm(BootstrapMixin, forms.ModelForm):
|
|
slug = SlugField()
|
|
content_types = ContentTypeMultipleChoiceField(
|
|
queryset=ContentType.objects.all()
|
|
)
|
|
parameters = JSONField()
|
|
|
|
fieldsets = (
|
|
('Saved Filter', ('name', 'slug', 'content_types', 'description', 'weight', 'enabled', 'shared')),
|
|
('Parameters', ('parameters',)),
|
|
)
|
|
|
|
class Meta:
|
|
model = SavedFilter
|
|
exclude = ('user',)
|
|
|
|
def __init__(self, *args, initial=None, **kwargs):
|
|
|
|
# Convert any parameters delivered via initial data to a dictionary
|
|
if initial and 'parameters' in initial:
|
|
if type(initial['parameters']) is str:
|
|
# TODO: Make a utility function for this
|
|
initial['parameters'] = dict(QueryDict(initial['parameters']).lists())
|
|
|
|
super().__init__(*args, initial=initial, **kwargs)
|
|
|
|
|
|
class WebhookForm(BootstrapMixin, forms.ModelForm):
|
|
content_types = ContentTypeMultipleChoiceField(
|
|
queryset=ContentType.objects.all(),
|
|
limit_choices_to=FeatureQuery('webhooks')
|
|
)
|
|
|
|
fieldsets = (
|
|
('Webhook', ('name', 'content_types', 'enabled')),
|
|
('Events', ('type_create', 'type_update', 'type_delete')),
|
|
('HTTP Request', (
|
|
'payload_url', 'http_method', 'http_content_type', 'additional_headers', 'body_template', 'secret',
|
|
)),
|
|
('Conditions', ('conditions',)),
|
|
('SSL', ('ssl_verification', 'ca_file_path')),
|
|
)
|
|
|
|
class Meta:
|
|
model = Webhook
|
|
fields = '__all__'
|
|
labels = {
|
|
'type_create': 'Creations',
|
|
'type_update': 'Updates',
|
|
'type_delete': 'Deletions',
|
|
}
|
|
widgets = {
|
|
'http_method': StaticSelect(),
|
|
'additional_headers': forms.Textarea(attrs={'class': 'font-monospace'}),
|
|
'body_template': forms.Textarea(attrs={'class': 'font-monospace'}),
|
|
'conditions': forms.Textarea(attrs={'class': 'font-monospace'}),
|
|
}
|
|
|
|
|
|
class TagForm(BootstrapMixin, forms.ModelForm):
|
|
slug = SlugField()
|
|
|
|
fieldsets = (
|
|
('Tag', ('name', 'slug', 'color', 'description')),
|
|
)
|
|
|
|
class Meta:
|
|
model = Tag
|
|
fields = [
|
|
'name', 'slug', 'color', 'description'
|
|
]
|
|
|
|
|
|
class ConfigContextForm(BootstrapMixin, SyncedDataMixin, forms.ModelForm):
|
|
regions = DynamicModelMultipleChoiceField(
|
|
queryset=Region.objects.all(),
|
|
required=False
|
|
)
|
|
site_groups = DynamicModelMultipleChoiceField(
|
|
queryset=SiteGroup.objects.all(),
|
|
required=False
|
|
)
|
|
sites = DynamicModelMultipleChoiceField(
|
|
queryset=Site.objects.all(),
|
|
required=False
|
|
)
|
|
locations = DynamicModelMultipleChoiceField(
|
|
queryset=Location.objects.all(),
|
|
required=False
|
|
)
|
|
device_types = DynamicModelMultipleChoiceField(
|
|
queryset=DeviceType.objects.all(),
|
|
required=False
|
|
)
|
|
roles = DynamicModelMultipleChoiceField(
|
|
queryset=DeviceRole.objects.all(),
|
|
required=False
|
|
)
|
|
platforms = DynamicModelMultipleChoiceField(
|
|
queryset=Platform.objects.all(),
|
|
required=False
|
|
)
|
|
cluster_types = DynamicModelMultipleChoiceField(
|
|
queryset=ClusterType.objects.all(),
|
|
required=False
|
|
)
|
|
cluster_groups = DynamicModelMultipleChoiceField(
|
|
queryset=ClusterGroup.objects.all(),
|
|
required=False
|
|
)
|
|
clusters = DynamicModelMultipleChoiceField(
|
|
queryset=Cluster.objects.all(),
|
|
required=False
|
|
)
|
|
tenant_groups = DynamicModelMultipleChoiceField(
|
|
queryset=TenantGroup.objects.all(),
|
|
required=False
|
|
)
|
|
tenants = DynamicModelMultipleChoiceField(
|
|
queryset=Tenant.objects.all(),
|
|
required=False
|
|
)
|
|
tags = DynamicModelMultipleChoiceField(
|
|
queryset=Tag.objects.all(),
|
|
required=False
|
|
)
|
|
data = JSONField(
|
|
required=False
|
|
)
|
|
|
|
fieldsets = (
|
|
('Config Context', ('name', 'weight', 'description', 'data', 'is_active')),
|
|
('Data Source', ('data_source', 'data_file')),
|
|
('Assignment', (
|
|
'regions', 'site_groups', 'sites', 'locations', 'device_types', 'roles', 'platforms', 'cluster_types',
|
|
'cluster_groups', 'clusters', 'tenant_groups', 'tenants', 'tags',
|
|
)),
|
|
)
|
|
|
|
class Meta:
|
|
model = ConfigContext
|
|
fields = (
|
|
'name', 'weight', 'description', 'data', 'is_active', 'regions', 'site_groups', 'sites', 'locations',
|
|
'roles', 'device_types', 'platforms', 'cluster_types', 'cluster_groups', 'clusters', 'tenant_groups',
|
|
'tenants', 'tags', 'data_source', 'data_file',
|
|
)
|
|
|
|
def clean(self):
|
|
super().clean()
|
|
|
|
if not self.cleaned_data.get('data') and not self.cleaned_data.get('data_source'):
|
|
raise forms.ValidationError("Must specify either local data or a data source")
|
|
|
|
return self.cleaned_data
|
|
|
|
|
|
class ImageAttachmentForm(BootstrapMixin, forms.ModelForm):
|
|
|
|
class Meta:
|
|
model = ImageAttachment
|
|
fields = [
|
|
'name', 'image',
|
|
]
|
|
|
|
|
|
class JournalEntryForm(NetBoxModelForm):
|
|
kind = forms.ChoiceField(
|
|
choices=add_blank_choice(JournalEntryKindChoices),
|
|
required=False,
|
|
widget=StaticSelect()
|
|
)
|
|
comments = CommentField()
|
|
|
|
class Meta:
|
|
model = JournalEntry
|
|
fields = ['assigned_object_type', 'assigned_object_id', 'kind', 'tags', 'comments']
|
|
widgets = {
|
|
'assigned_object_type': forms.HiddenInput,
|
|
'assigned_object_id': forms.HiddenInput,
|
|
}
|