| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078 |
- from django.contrib import messages
- from django.contrib.auth.mixins import LoginRequiredMixin
- from django.contrib.contenttypes.models import ContentType
- from django.db.models import Count, Q
- from django.http import Http404, HttpResponseForbidden, HttpResponse
- from django.shortcuts import get_object_or_404, redirect, render
- from django.urls import reverse
- from django.views.generic import View
- from django_rq.queues import get_connection
- from rq import Worker
- from extras.dashboard.forms import DashboardWidgetAddForm, DashboardWidgetForm
- from extras.dashboard.utils import get_widget_class
- from netbox.views import generic
- from utilities.forms import ConfirmationForm, get_field_value
- from utilities.htmx import is_htmx
- from utilities.utils import copy_safe_request, count_related, get_viewname, normalize_querydict, shallow_compare_dict
- from utilities.views import ContentTypePermissionRequiredMixin, register_model_view
- from . import filtersets, forms, tables
- from .choices import JobResultStatusChoices
- from .forms.reports import ReportForm
- from .models import *
- from .reports import get_report, get_reports, run_report
- from .scripts import get_scripts, run_script
- #
- # Custom fields
- #
- class CustomFieldListView(generic.ObjectListView):
- queryset = CustomField.objects.all()
- filterset = filtersets.CustomFieldFilterSet
- filterset_form = forms.CustomFieldFilterForm
- table = tables.CustomFieldTable
- @register_model_view(CustomField)
- class CustomFieldView(generic.ObjectView):
- queryset = CustomField.objects.all()
- @register_model_view(CustomField, 'edit')
- class CustomFieldEditView(generic.ObjectEditView):
- queryset = CustomField.objects.all()
- form = forms.CustomFieldForm
- @register_model_view(CustomField, 'delete')
- class CustomFieldDeleteView(generic.ObjectDeleteView):
- queryset = CustomField.objects.all()
- class CustomFieldBulkImportView(generic.BulkImportView):
- queryset = CustomField.objects.all()
- model_form = forms.CustomFieldImportForm
- table = tables.CustomFieldTable
- class CustomFieldBulkEditView(generic.BulkEditView):
- queryset = CustomField.objects.all()
- filterset = filtersets.CustomFieldFilterSet
- table = tables.CustomFieldTable
- form = forms.CustomFieldBulkEditForm
- class CustomFieldBulkDeleteView(generic.BulkDeleteView):
- queryset = CustomField.objects.all()
- filterset = filtersets.CustomFieldFilterSet
- table = tables.CustomFieldTable
- #
- # Custom links
- #
- class CustomLinkListView(generic.ObjectListView):
- queryset = CustomLink.objects.all()
- filterset = filtersets.CustomLinkFilterSet
- filterset_form = forms.CustomLinkFilterForm
- table = tables.CustomLinkTable
- @register_model_view(CustomLink)
- class CustomLinkView(generic.ObjectView):
- queryset = CustomLink.objects.all()
- @register_model_view(CustomLink, 'edit')
- class CustomLinkEditView(generic.ObjectEditView):
- queryset = CustomLink.objects.all()
- form = forms.CustomLinkForm
- @register_model_view(CustomLink, 'delete')
- class CustomLinkDeleteView(generic.ObjectDeleteView):
- queryset = CustomLink.objects.all()
- class CustomLinkBulkImportView(generic.BulkImportView):
- queryset = CustomLink.objects.all()
- model_form = forms.CustomLinkImportForm
- table = tables.CustomLinkTable
- class CustomLinkBulkEditView(generic.BulkEditView):
- queryset = CustomLink.objects.all()
- filterset = filtersets.CustomLinkFilterSet
- table = tables.CustomLinkTable
- form = forms.CustomLinkBulkEditForm
- class CustomLinkBulkDeleteView(generic.BulkDeleteView):
- queryset = CustomLink.objects.all()
- filterset = filtersets.CustomLinkFilterSet
- table = tables.CustomLinkTable
- #
- # Export templates
- #
- class ExportTemplateListView(generic.ObjectListView):
- queryset = ExportTemplate.objects.all()
- filterset = filtersets.ExportTemplateFilterSet
- filterset_form = forms.ExportTemplateFilterForm
- table = tables.ExportTemplateTable
- template_name = 'extras/exporttemplate_list.html'
- actions = ('add', 'import', 'export', 'bulk_edit', 'bulk_delete', 'bulk_sync')
- @register_model_view(ExportTemplate)
- class ExportTemplateView(generic.ObjectView):
- queryset = ExportTemplate.objects.all()
- @register_model_view(ExportTemplate, 'edit')
- class ExportTemplateEditView(generic.ObjectEditView):
- queryset = ExportTemplate.objects.all()
- form = forms.ExportTemplateForm
- @register_model_view(ExportTemplate, 'delete')
- class ExportTemplateDeleteView(generic.ObjectDeleteView):
- queryset = ExportTemplate.objects.all()
- class ExportTemplateBulkImportView(generic.BulkImportView):
- queryset = ExportTemplate.objects.all()
- model_form = forms.ExportTemplateImportForm
- table = tables.ExportTemplateTable
- class ExportTemplateBulkEditView(generic.BulkEditView):
- queryset = ExportTemplate.objects.all()
- filterset = filtersets.ExportTemplateFilterSet
- table = tables.ExportTemplateTable
- form = forms.ExportTemplateBulkEditForm
- class ExportTemplateBulkDeleteView(generic.BulkDeleteView):
- queryset = ExportTemplate.objects.all()
- filterset = filtersets.ExportTemplateFilterSet
- table = tables.ExportTemplateTable
- class ExportTemplateBulkSyncDataView(generic.BulkSyncDataView):
- queryset = ExportTemplate.objects.all()
- #
- # Saved filters
- #
- class SavedFilterMixin:
- def get_queryset(self, request):
- """
- Return only shared SavedFilters, or those owned by the current user, unless
- this is a superuser.
- """
- queryset = SavedFilter.objects.all()
- user = request.user
- if user.is_superuser:
- return queryset
- if user.is_anonymous:
- return queryset.filter(shared=True)
- return queryset.filter(
- Q(shared=True) | Q(user=user)
- )
- class SavedFilterListView(SavedFilterMixin, generic.ObjectListView):
- filterset = filtersets.SavedFilterFilterSet
- filterset_form = forms.SavedFilterFilterForm
- table = tables.SavedFilterTable
- @register_model_view(SavedFilter)
- class SavedFilterView(SavedFilterMixin, generic.ObjectView):
- queryset = SavedFilter.objects.all()
- @register_model_view(SavedFilter, 'edit')
- class SavedFilterEditView(SavedFilterMixin, generic.ObjectEditView):
- queryset = SavedFilter.objects.all()
- form = forms.SavedFilterForm
- def alter_object(self, obj, request, url_args, url_kwargs):
- if not obj.pk:
- obj.user = request.user
- return obj
- @register_model_view(SavedFilter, 'delete')
- class SavedFilterDeleteView(SavedFilterMixin, generic.ObjectDeleteView):
- queryset = SavedFilter.objects.all()
- class SavedFilterBulkImportView(SavedFilterMixin, generic.BulkImportView):
- queryset = SavedFilter.objects.all()
- model_form = forms.SavedFilterImportForm
- table = tables.SavedFilterTable
- class SavedFilterBulkEditView(SavedFilterMixin, generic.BulkEditView):
- queryset = SavedFilter.objects.all()
- filterset = filtersets.SavedFilterFilterSet
- table = tables.SavedFilterTable
- form = forms.SavedFilterBulkEditForm
- class SavedFilterBulkDeleteView(SavedFilterMixin, generic.BulkDeleteView):
- queryset = SavedFilter.objects.all()
- filterset = filtersets.SavedFilterFilterSet
- table = tables.SavedFilterTable
- #
- # Webhooks
- #
- class WebhookListView(generic.ObjectListView):
- queryset = Webhook.objects.all()
- filterset = filtersets.WebhookFilterSet
- filterset_form = forms.WebhookFilterForm
- table = tables.WebhookTable
- @register_model_view(Webhook)
- class WebhookView(generic.ObjectView):
- queryset = Webhook.objects.all()
- @register_model_view(Webhook, 'edit')
- class WebhookEditView(generic.ObjectEditView):
- queryset = Webhook.objects.all()
- form = forms.WebhookForm
- @register_model_view(Webhook, 'delete')
- class WebhookDeleteView(generic.ObjectDeleteView):
- queryset = Webhook.objects.all()
- class WebhookBulkImportView(generic.BulkImportView):
- queryset = Webhook.objects.all()
- model_form = forms.WebhookImportForm
- table = tables.WebhookTable
- class WebhookBulkEditView(generic.BulkEditView):
- queryset = Webhook.objects.all()
- filterset = filtersets.WebhookFilterSet
- table = tables.WebhookTable
- form = forms.WebhookBulkEditForm
- class WebhookBulkDeleteView(generic.BulkDeleteView):
- queryset = Webhook.objects.all()
- filterset = filtersets.WebhookFilterSet
- table = tables.WebhookTable
- #
- # Tags
- #
- class TagListView(generic.ObjectListView):
- queryset = Tag.objects.annotate(
- items=count_related(TaggedItem, 'tag')
- )
- filterset = filtersets.TagFilterSet
- filterset_form = forms.TagFilterForm
- table = tables.TagTable
- @register_model_view(Tag)
- class TagView(generic.ObjectView):
- queryset = Tag.objects.all()
- def get_extra_context(self, request, instance):
- tagged_items = TaggedItem.objects.filter(tag=instance)
- taggeditem_table = tables.TaggedItemTable(
- data=tagged_items,
- orderable=False
- )
- taggeditem_table.configure(request)
- object_types = [
- {
- 'content_type': ContentType.objects.get(pk=ti['content_type']),
- 'item_count': ti['item_count']
- } for ti in tagged_items.values('content_type').annotate(item_count=Count('pk'))
- ]
- return {
- 'taggeditem_table': taggeditem_table,
- 'tagged_item_count': tagged_items.count(),
- 'object_types': object_types,
- }
- @register_model_view(Tag, 'edit')
- class TagEditView(generic.ObjectEditView):
- queryset = Tag.objects.all()
- form = forms.TagForm
- @register_model_view(Tag, 'delete')
- class TagDeleteView(generic.ObjectDeleteView):
- queryset = Tag.objects.all()
- class TagBulkImportView(generic.BulkImportView):
- queryset = Tag.objects.all()
- model_form = forms.TagImportForm
- table = tables.TagTable
- class TagBulkEditView(generic.BulkEditView):
- queryset = Tag.objects.annotate(
- items=count_related(TaggedItem, 'tag')
- )
- table = tables.TagTable
- form = forms.TagBulkEditForm
- class TagBulkDeleteView(generic.BulkDeleteView):
- queryset = Tag.objects.annotate(
- items=count_related(TaggedItem, 'tag')
- )
- table = tables.TagTable
- #
- # Config contexts
- #
- class ConfigContextListView(generic.ObjectListView):
- queryset = ConfigContext.objects.all()
- filterset = filtersets.ConfigContextFilterSet
- filterset_form = forms.ConfigContextFilterForm
- table = tables.ConfigContextTable
- template_name = 'extras/configcontext_list.html'
- actions = ('add', 'bulk_edit', 'bulk_delete', 'bulk_sync')
- @register_model_view(ConfigContext)
- class ConfigContextView(generic.ObjectView):
- queryset = ConfigContext.objects.all()
- def get_extra_context(self, request, instance):
- # Gather assigned objects for parsing in the template
- assigned_objects = (
- ('Regions', instance.regions.all),
- ('Site Groups', instance.site_groups.all),
- ('Sites', instance.sites.all),
- ('Locations', instance.locations.all),
- ('Device Types', instance.device_types.all),
- ('Roles', instance.roles.all),
- ('Platforms', instance.platforms.all),
- ('Cluster Types', instance.cluster_types.all),
- ('Cluster Groups', instance.cluster_groups.all),
- ('Clusters', instance.clusters.all),
- ('Tenant Groups', instance.tenant_groups.all),
- ('Tenants', instance.tenants.all),
- ('Tags', instance.tags.all),
- )
- # Determine user's preferred output format
- if request.GET.get('format') in ['json', 'yaml']:
- format = request.GET.get('format')
- if request.user.is_authenticated:
- request.user.config.set('data_format', format, commit=True)
- elif request.user.is_authenticated:
- format = request.user.config.get('data_format', 'json')
- else:
- format = 'json'
- return {
- 'assigned_objects': assigned_objects,
- 'format': format,
- }
- @register_model_view(ConfigContext, 'edit')
- class ConfigContextEditView(generic.ObjectEditView):
- queryset = ConfigContext.objects.all()
- form = forms.ConfigContextForm
- class ConfigContextBulkEditView(generic.BulkEditView):
- queryset = ConfigContext.objects.all()
- filterset = filtersets.ConfigContextFilterSet
- table = tables.ConfigContextTable
- form = forms.ConfigContextBulkEditForm
- @register_model_view(ConfigContext, 'delete')
- class ConfigContextDeleteView(generic.ObjectDeleteView):
- queryset = ConfigContext.objects.all()
- class ConfigContextBulkDeleteView(generic.BulkDeleteView):
- queryset = ConfigContext.objects.all()
- table = tables.ConfigContextTable
- class ConfigContextBulkSyncDataView(generic.BulkSyncDataView):
- queryset = ConfigContext.objects.all()
- class ObjectConfigContextView(generic.ObjectView):
- base_template = None
- template_name = 'extras/object_configcontext.html'
- def get_extra_context(self, request, instance):
- source_contexts = ConfigContext.objects.restrict(request.user, 'view').get_for_object(instance)
- # Determine user's preferred output format
- if request.GET.get('format') in ['json', 'yaml']:
- format = request.GET.get('format')
- if request.user.is_authenticated:
- request.user.config.set('data_format', format, commit=True)
- elif request.user.is_authenticated:
- format = request.user.config.get('data_format', 'json')
- else:
- format = 'json'
- return {
- 'rendered_context': instance.get_config_context(),
- 'source_contexts': source_contexts,
- 'format': format,
- 'base_template': self.base_template,
- }
- #
- # Config templates
- #
- class ConfigTemplateListView(generic.ObjectListView):
- queryset = ConfigTemplate.objects.all()
- filterset = filtersets.ConfigTemplateFilterSet
- filterset_form = forms.ConfigTemplateFilterForm
- table = tables.ConfigTemplateTable
- template_name = 'extras/configtemplate_list.html'
- actions = ('add', 'import', 'export', 'bulk_edit', 'bulk_delete', 'bulk_sync')
- @register_model_view(ConfigTemplate)
- class ConfigTemplateView(generic.ObjectView):
- queryset = ConfigTemplate.objects.all()
- @register_model_view(ConfigTemplate, 'edit')
- class ConfigTemplateEditView(generic.ObjectEditView):
- queryset = ConfigTemplate.objects.all()
- form = forms.ConfigTemplateForm
- @register_model_view(ConfigTemplate, 'delete')
- class ConfigTemplateDeleteView(generic.ObjectDeleteView):
- queryset = ConfigTemplate.objects.all()
- class ConfigTemplateBulkImportView(generic.BulkImportView):
- queryset = ConfigTemplate.objects.all()
- model_form = forms.ConfigTemplateImportForm
- table = tables.ConfigTemplateTable
- class ConfigTemplateBulkEditView(generic.BulkEditView):
- queryset = ConfigTemplate.objects.all()
- filterset = filtersets.ConfigTemplateFilterSet
- table = tables.ConfigTemplateTable
- form = forms.ConfigTemplateBulkEditForm
- class ConfigTemplateBulkDeleteView(generic.BulkDeleteView):
- queryset = ConfigTemplate.objects.all()
- filterset = filtersets.ConfigTemplateFilterSet
- table = tables.ConfigTemplateTable
- class ConfigTemplateBulkSyncDataView(generic.BulkSyncDataView):
- queryset = ConfigTemplate.objects.all()
- #
- # Change logging
- #
- class ObjectChangeListView(generic.ObjectListView):
- queryset = ObjectChange.objects.all()
- filterset = filtersets.ObjectChangeFilterSet
- filterset_form = forms.ObjectChangeFilterForm
- table = tables.ObjectChangeTable
- template_name = 'extras/objectchange_list.html'
- actions = ('export',)
- @register_model_view(ObjectChange)
- class ObjectChangeView(generic.ObjectView):
- queryset = ObjectChange.objects.all()
- def get_extra_context(self, request, instance):
- related_changes = ObjectChange.objects.restrict(request.user, 'view').filter(
- request_id=instance.request_id
- ).exclude(
- pk=instance.pk
- )
- related_changes_table = tables.ObjectChangeTable(
- data=related_changes[:50],
- orderable=False
- )
- objectchanges = ObjectChange.objects.restrict(request.user, 'view').filter(
- changed_object_type=instance.changed_object_type,
- changed_object_id=instance.changed_object_id,
- )
- next_change = objectchanges.filter(time__gt=instance.time).order_by('time').first()
- prev_change = objectchanges.filter(time__lt=instance.time).order_by('-time').first()
- if not instance.prechange_data and instance.action in ['update', 'delete'] and prev_change:
- non_atomic_change = True
- prechange_data = prev_change.postchange_data
- else:
- non_atomic_change = False
- prechange_data = instance.prechange_data
- if prechange_data and instance.postchange_data:
- diff_added = shallow_compare_dict(
- prechange_data or dict(),
- instance.postchange_data or dict(),
- exclude=['last_updated'],
- )
- diff_removed = {
- x: prechange_data.get(x) for x in diff_added
- } if prechange_data else {}
- else:
- diff_added = None
- diff_removed = None
- return {
- 'diff_added': diff_added,
- 'diff_removed': diff_removed,
- 'next_change': next_change,
- 'prev_change': prev_change,
- 'related_changes_table': related_changes_table,
- 'related_changes_count': related_changes.count(),
- 'non_atomic_change': non_atomic_change
- }
- #
- # Image attachments
- #
- @register_model_view(ImageAttachment, 'edit')
- class ImageAttachmentEditView(generic.ObjectEditView):
- queryset = ImageAttachment.objects.all()
- form = forms.ImageAttachmentForm
- template_name = 'extras/imageattachment_edit.html'
- def alter_object(self, instance, request, args, kwargs):
- if not instance.pk:
- # Assign the parent object based on URL kwargs
- content_type = get_object_or_404(ContentType, pk=request.GET.get('content_type'))
- instance.parent = get_object_or_404(content_type.model_class(), pk=request.GET.get('object_id'))
- return instance
- def get_return_url(self, request, obj=None):
- return obj.parent.get_absolute_url() if obj else super().get_return_url(request)
- def get_extra_addanother_params(self, request):
- return {
- 'content_type': request.GET.get('content_type'),
- 'object_id': request.GET.get('object_id'),
- }
- @register_model_view(ImageAttachment, 'delete')
- class ImageAttachmentDeleteView(generic.ObjectDeleteView):
- queryset = ImageAttachment.objects.all()
- def get_return_url(self, request, obj=None):
- return obj.parent.get_absolute_url() if obj else super().get_return_url(request)
- #
- # Journal entries
- #
- class JournalEntryListView(generic.ObjectListView):
- queryset = JournalEntry.objects.all()
- filterset = filtersets.JournalEntryFilterSet
- filterset_form = forms.JournalEntryFilterForm
- table = tables.JournalEntryTable
- actions = ('export', 'bulk_edit', 'bulk_delete')
- @register_model_view(JournalEntry)
- class JournalEntryView(generic.ObjectView):
- queryset = JournalEntry.objects.all()
- @register_model_view(JournalEntry, 'edit')
- class JournalEntryEditView(generic.ObjectEditView):
- queryset = JournalEntry.objects.all()
- form = forms.JournalEntryForm
- def alter_object(self, obj, request, args, kwargs):
- if not obj.pk:
- obj.created_by = request.user
- return obj
- def get_return_url(self, request, instance):
- if not instance.assigned_object:
- return reverse('extras:journalentry_list')
- obj = instance.assigned_object
- viewname = get_viewname(obj, 'journal')
- return reverse(viewname, kwargs={'pk': obj.pk})
- @register_model_view(JournalEntry, 'delete')
- class JournalEntryDeleteView(generic.ObjectDeleteView):
- queryset = JournalEntry.objects.all()
- def get_return_url(self, request, instance):
- obj = instance.assigned_object
- viewname = get_viewname(obj, 'journal')
- return reverse(viewname, kwargs={'pk': obj.pk})
- class JournalEntryBulkEditView(generic.BulkEditView):
- queryset = JournalEntry.objects.all()
- filterset = filtersets.JournalEntryFilterSet
- table = tables.JournalEntryTable
- form = forms.JournalEntryBulkEditForm
- class JournalEntryBulkDeleteView(generic.BulkDeleteView):
- queryset = JournalEntry.objects.all()
- filterset = filtersets.JournalEntryFilterSet
- table = tables.JournalEntryTable
- #
- # Dashboard widgets
- #
- class DashboardWidgetAddView(LoginRequiredMixin, View):
- template_name = 'extras/dashboard/widget_add.html'
- def get(self, request):
- if not is_htmx(request):
- return redirect('home')
- initial = request.GET or {
- 'widget_class': 'extras.NoteWidget',
- }
- widget_form = DashboardWidgetAddForm(initial=initial)
- widget_name = get_field_value(widget_form, 'widget_class')
- widget_class = get_widget_class(widget_name)
- config_form = widget_class.ConfigForm(prefix='config')
- return render(request, self.template_name, {
- 'widget_class': widget_class,
- 'widget_form': widget_form,
- 'config_form': config_form,
- })
- def post(self, request):
- widget_form = DashboardWidgetAddForm(request.POST)
- config_form = None
- widget_class = None
- if widget_form.is_valid():
- widget_class = get_widget_class(widget_form.cleaned_data['widget_class'])
- config_form = widget_class.ConfigForm(request.POST, prefix='config')
- if config_form.is_valid():
- data = widget_form.cleaned_data
- data.pop('widget_class')
- data['config'] = config_form.cleaned_data
- widget = widget_class(**data)
- request.user.dashboard.add_widget(widget)
- request.user.dashboard.save()
- messages.success(request, f'Added widget {widget.id}')
- return HttpResponse(headers={
- 'HX-Redirect': reverse('home'),
- })
- return render(request, self.template_name, {
- 'widget_class': widget_class,
- 'widget_form': widget_form,
- 'config_form': config_form,
- })
- class DashboardWidgetConfigView(LoginRequiredMixin, View):
- template_name = 'extras/dashboard/widget_config.html'
- def get(self, request, id):
- if not is_htmx(request):
- return redirect('home')
- widget = request.user.dashboard.get_widget(id)
- widget_form = DashboardWidgetForm(initial=widget.form_data)
- config_form = widget.ConfigForm(initial=widget.form_data.get('config'), prefix='config')
- return render(request, self.template_name, {
- 'widget_form': widget_form,
- 'config_form': config_form,
- 'form_url': reverse('extras:dashboardwidget_config', kwargs={'id': id})
- })
- def post(self, request, id):
- widget = request.user.dashboard.get_widget(id)
- widget_form = DashboardWidgetForm(request.POST)
- config_form = widget.ConfigForm(request.POST, prefix='config')
- if widget_form.is_valid() and config_form.is_valid():
- data = widget_form.cleaned_data
- data['config'] = config_form.cleaned_data
- request.user.dashboard.config[str(id)].update(data)
- request.user.dashboard.save()
- messages.success(request, f'Updated widget {widget.id}')
- return HttpResponse(headers={
- 'HX-Redirect': reverse('home'),
- })
- return render(request, self.template_name, {
- 'widget_form': widget_form,
- 'config_form': config_form,
- 'form_url': reverse('extras:dashboardwidget_config', kwargs={'id': id})
- })
- class DashboardWidgetDeleteView(LoginRequiredMixin, View):
- template_name = 'generic/object_delete.html'
- def get(self, request, id):
- if not is_htmx(request):
- return redirect('home')
- widget = request.user.dashboard.get_widget(id)
- form = ConfirmationForm(initial=request.GET)
- return render(request, 'htmx/delete_form.html', {
- 'object_type': widget.__class__.__name__,
- 'object': widget,
- 'form': form,
- 'form_url': reverse('extras:dashboardwidget_delete', kwargs={'id': id})
- })
- def post(self, request, id):
- form = ConfirmationForm(request.POST)
- if form.is_valid():
- request.user.dashboard.delete_widget(id)
- request.user.dashboard.save()
- messages.success(request, f'Deleted widget {id}')
- else:
- messages.error(request, f'Error deleting widget: {form.errors[0]}')
- return redirect(reverse('home'))
- #
- # Reports
- #
- class ReportListView(ContentTypePermissionRequiredMixin, View):
- """
- Retrieve all of the available reports from disk and the recorded JobResult (if any) for each.
- """
- def get_required_permission(self):
- return 'extras.view_report'
- def get(self, request):
- reports = get_reports()
- report_content_type = ContentType.objects.get(app_label='extras', model='report')
- results = {
- r.name: r
- for r in JobResult.objects.filter(
- obj_type=report_content_type,
- status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
- ).order_by('name', '-created').distinct('name').defer('data')
- }
- ret = []
- for module, report_list in reports.items():
- module_reports = []
- for report in report_list.values():
- report.result = results.get(report.full_name, None)
- module_reports.append(report)
- ret.append((module, module_reports))
- return render(request, 'extras/report_list.html', {
- 'reports': ret,
- })
- class ReportView(ContentTypePermissionRequiredMixin, View):
- """
- Display a single Report and its associated JobResult (if any).
- """
- def get_required_permission(self):
- return 'extras.view_report'
- def get(self, request, module, name):
- report = get_report(module, name)
- if report is None:
- raise Http404
- report_content_type = ContentType.objects.get(app_label='extras', model='report')
- report.result = JobResult.objects.filter(
- obj_type=report_content_type,
- name=report.full_name,
- status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
- ).first()
- return render(request, 'extras/report.html', {
- 'report': report,
- 'form': ReportForm(),
- })
- def post(self, request, module, name):
- # Permissions check
- if not request.user.has_perm('extras.run_report'):
- return HttpResponseForbidden()
- report = get_report(module, name)
- if report is None:
- raise Http404
- form = ReportForm(request.POST)
- if form.is_valid():
- # Allow execution only if RQ worker process is running
- if not Worker.count(get_connection('default')):
- messages.error(request, "Unable to run report: RQ worker process not running.")
- return render(request, 'extras/report.html', {
- 'report': report,
- })
- # Run the Report. A new JobResult is created.
- job_result = JobResult.enqueue_job(
- run_report,
- name=report.full_name,
- obj_type=ContentType.objects.get_for_model(Report),
- user=request.user,
- schedule_at=form.cleaned_data.get('schedule_at'),
- interval=form.cleaned_data.get('interval'),
- job_timeout=report.job_timeout
- )
- return redirect('extras:report_result', job_result_pk=job_result.pk)
- return render(request, 'extras/report.html', {
- 'report': report,
- 'form': form,
- })
- class ReportResultView(ContentTypePermissionRequiredMixin, View):
- """
- Display a JobResult pertaining to the execution of a Report.
- """
- def get_required_permission(self):
- return 'extras.view_report'
- def get(self, request, job_result_pk):
- report_content_type = ContentType.objects.get(app_label='extras', model='report')
- result = get_object_or_404(JobResult.objects.all(), pk=job_result_pk, obj_type=report_content_type)
- # Retrieve the Report and attach the JobResult to it
- module, report_name = result.name.split('.', maxsplit=1)
- report = get_report(module, report_name)
- report.result = result
- # If this is an HTMX request, return only the result HTML
- if is_htmx(request):
- response = render(request, 'extras/htmx/report_result.html', {
- 'report': report,
- 'result': result,
- })
- if result.completed or not result.started:
- response.status_code = 286
- return response
- return render(request, 'extras/report_result.html', {
- 'report': report,
- 'result': result,
- })
- #
- # Scripts
- #
- class GetScriptMixin:
- def _get_script(self, name, module=None):
- if module is None:
- module, name = name.split('.', 1)
- scripts = get_scripts()
- try:
- return scripts[module][name]()
- except KeyError:
- raise Http404
- class ScriptListView(ContentTypePermissionRequiredMixin, View):
- def get_required_permission(self):
- return 'extras.view_script'
- def get(self, request):
- scripts = get_scripts(use_names=True)
- script_content_type = ContentType.objects.get(app_label='extras', model='script')
- results = {
- r.name: r
- for r in JobResult.objects.filter(
- obj_type=script_content_type,
- status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
- ).order_by('name', '-created').distinct('name').defer('data')
- }
- for _scripts in scripts.values():
- for script in _scripts.values():
- script.result = results.get(script.full_name)
- return render(request, 'extras/script_list.html', {
- 'scripts': scripts,
- })
- class ScriptView(ContentTypePermissionRequiredMixin, GetScriptMixin, View):
- def get_required_permission(self):
- return 'extras.view_script'
- def get(self, request, module, name):
- script = self._get_script(name, module)
- form = script.as_form(initial=normalize_querydict(request.GET))
- # Look for a pending JobResult (use the latest one by creation timestamp)
- script.result = JobResult.objects.filter(
- obj_type=ContentType.objects.get_for_model(Script),
- name=script.full_name,
- ).exclude(
- status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
- ).first()
- return render(request, 'extras/script.html', {
- 'module': module,
- 'script': script,
- 'form': form,
- })
- def post(self, request, module, name):
- # Permissions check
- if not request.user.has_perm('extras.run_script'):
- return HttpResponseForbidden()
- script = self._get_script(name, module)
- form = script.as_form(request.POST, request.FILES)
- # Allow execution only if RQ worker process is running
- if not Worker.count(get_connection('default')):
- messages.error(request, "Unable to run script: RQ worker process not running.")
- elif form.is_valid():
- job_result = JobResult.enqueue_job(
- run_script,
- name=script.full_name,
- obj_type=ContentType.objects.get_for_model(Script),
- user=request.user,
- schedule_at=form.cleaned_data.pop('_schedule_at'),
- interval=form.cleaned_data.pop('_interval'),
- data=form.cleaned_data,
- request=copy_safe_request(request),
- job_timeout=script.job_timeout,
- commit=form.cleaned_data.pop('_commit')
- )
- return redirect('extras:script_result', job_result_pk=job_result.pk)
- return render(request, 'extras/script.html', {
- 'module': module,
- 'script': script,
- 'form': form,
- })
- class ScriptResultView(ContentTypePermissionRequiredMixin, GetScriptMixin, View):
- def get_required_permission(self):
- return 'extras.view_script'
- def get(self, request, job_result_pk):
- result = get_object_or_404(JobResult.objects.all(), pk=job_result_pk)
- script_content_type = ContentType.objects.get(app_label='extras', model='script')
- if result.obj_type != script_content_type:
- raise Http404
- script = self._get_script(result.name)
- # If this is an HTMX request, return only the result HTML
- if is_htmx(request):
- response = render(request, 'extras/htmx/script_result.html', {
- 'script': script,
- 'result': result,
- })
- if result.completed or not result.started:
- response.status_code = 286
- return response
- return render(request, 'extras/script_result.html', {
- 'script': script,
- 'result': result,
- 'class_name': script.__class__.__name__
- })
- #
- # Job results
- #
- class JobResultListView(generic.ObjectListView):
- queryset = JobResult.objects.all()
- filterset = filtersets.JobResultFilterSet
- filterset_form = forms.JobResultFilterForm
- table = tables.JobResultTable
- actions = ('export', 'delete', 'bulk_delete', )
- class JobResultDeleteView(generic.ObjectDeleteView):
- queryset = JobResult.objects.all()
- class JobResultBulkDeleteView(generic.BulkDeleteView):
- queryset = JobResult.objects.all()
- filterset = filtersets.JobResultFilterSet
- table = tables.JobResultTable
|