1
0
mirror of https://github.com/netbox-community/netbox.git synced 2024-05-10 07:54:54 +00:00

758 lines
23 KiB
Python
Raw Normal View History

from django.contrib import messages
2018-06-14 16:15:14 -04:00
from django.contrib.contenttypes.models import ContentType
2021-04-05 13:53:25 -04:00
from django.db.models import Count, Q
2019-08-12 11:39:36 -04:00
from django.http import Http404, HttpResponseForbidden
2018-07-03 09:47:44 -04:00
from django.shortcuts import get_object_or_404, redirect, render
from django.urls import reverse
2017-09-21 16:32:05 -04:00
from django.views.generic import View
from django_rq.queues import get_connection
from rq import Worker
2020-11-11 16:07:38 -05:00
from netbox.views import generic
2017-09-28 12:51:10 -04:00
from utilities.forms import ConfirmationForm
2021-12-20 20:45:32 -05:00
from utilities.htmx import is_htmx
from utilities.utils import copy_safe_request, count_related, get_viewname, normalize_querydict, shallow_compare_dict
2020-11-11 16:07:38 -05:00
from utilities.views import ContentTypePermissionRequiredMixin
2021-04-29 15:59:11 -04:00
from . import filtersets, forms, tables
from .choices import JobResultStatusChoices
2021-06-22 16:28:06 -04:00
from .models import *
from .reports import get_report, get_reports, run_report
2019-08-12 13:51:25 -04:00
from .scripts import get_scripts, run_script
2018-05-22 12:22:46 -04:00
2021-06-22 16:28:06 -04:00
#
# Custom fields
#
class CustomFieldListView(generic.ObjectListView):
queryset = CustomField.objects.all()
filterset = filtersets.CustomFieldFilterSet
filterset_form = forms.CustomFieldFilterForm
table = tables.CustomFieldTable
class CustomFieldView(generic.ObjectView):
queryset = CustomField.objects.all()
class CustomFieldEditView(generic.ObjectEditView):
queryset = CustomField.objects.all()
model_form = forms.CustomFieldForm
class CustomFieldDeleteView(generic.ObjectDeleteView):
queryset = CustomField.objects.all()
class CustomFieldBulkImportView(generic.BulkImportView):
queryset = CustomField.objects.all()
model_form = forms.CustomFieldCSVForm
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
2021-06-23 17:09:15 -04:00
#
# Custom links
#
class CustomLinkListView(generic.ObjectListView):
queryset = CustomLink.objects.all()
filterset = filtersets.CustomLinkFilterSet
filterset_form = forms.CustomLinkFilterForm
table = tables.CustomLinkTable
class CustomLinkView(generic.ObjectView):
queryset = CustomLink.objects.all()
class CustomLinkEditView(generic.ObjectEditView):
queryset = CustomLink.objects.all()
model_form = forms.CustomLinkForm
class CustomLinkDeleteView(generic.ObjectDeleteView):
queryset = CustomLink.objects.all()
class CustomLinkBulkImportView(generic.BulkImportView):
queryset = CustomLink.objects.all()
model_form = forms.CustomLinkCSVForm
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
2021-06-23 20:39:35 -04:00
#
# Export templates
#
class ExportTemplateListView(generic.ObjectListView):
queryset = ExportTemplate.objects.all()
filterset = filtersets.ExportTemplateFilterSet
filterset_form = forms.ExportTemplateFilterForm
table = tables.ExportTemplateTable
class ExportTemplateView(generic.ObjectView):
queryset = ExportTemplate.objects.all()
class ExportTemplateEditView(generic.ObjectEditView):
queryset = ExportTemplate.objects.all()
model_form = forms.ExportTemplateForm
class ExportTemplateDeleteView(generic.ObjectDeleteView):
queryset = ExportTemplate.objects.all()
class ExportTemplateBulkImportView(generic.BulkImportView):
queryset = ExportTemplate.objects.all()
model_form = forms.ExportTemplateCSVForm
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
2021-06-23 21:24:23 -04:00
#
# Webhooks
#
class WebhookListView(generic.ObjectListView):
queryset = Webhook.objects.all()
filterset = filtersets.WebhookFilterSet
filterset_form = forms.WebhookFilterForm
table = tables.WebhookTable
class WebhookView(generic.ObjectView):
queryset = Webhook.objects.all()
class WebhookEditView(generic.ObjectEditView):
queryset = Webhook.objects.all()
model_form = forms.WebhookForm
class WebhookDeleteView(generic.ObjectDeleteView):
queryset = Webhook.objects.all()
class WebhookBulkImportView(generic.BulkImportView):
queryset = Webhook.objects.all()
model_form = forms.WebhookCSVForm
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
2018-05-22 12:22:46 -04:00
#
# Tags
#
2020-11-11 16:07:38 -05:00
class TagListView(generic.ObjectListView):
queryset = Tag.objects.annotate(
items=count_related(TaggedItem, 'tag')
)
2021-04-29 15:59:11 -04:00
filterset = filtersets.TagFilterSet
filterset_form = forms.TagFilterForm
table = tables.TagTable
2018-05-22 12:22:46 -04:00
2021-03-26 15:25:18 -04:00
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)
2021-03-26 15:25:18 -04:00
2021-04-05 13:53:25 -04:00
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'))
]
2021-03-26 15:25:18 -04:00
return {
'taggeditem_table': taggeditem_table,
2021-03-26 15:25:18 -04:00
'tagged_item_count': tagged_items.count(),
2021-04-05 13:53:25 -04:00
'object_types': object_types,
2021-03-26 15:25:18 -04:00
}
2020-11-11 16:07:38 -05:00
class TagEditView(generic.ObjectEditView):
queryset = Tag.objects.all()
model_form = forms.TagForm
2018-05-22 12:22:46 -04:00
2020-11-11 16:07:38 -05:00
class TagDeleteView(generic.ObjectDeleteView):
queryset = Tag.objects.all()
2018-05-22 12:22:46 -04:00
2020-11-11 16:07:38 -05:00
class TagBulkImportView(generic.BulkImportView):
queryset = Tag.objects.all()
model_form = forms.TagCSVForm
table = tables.TagTable
2020-11-11 16:07:38 -05:00
class TagBulkEditView(generic.BulkEditView):
queryset = Tag.objects.annotate(
items=count_related(TaggedItem, 'tag')
)
table = tables.TagTable
form = forms.TagBulkEditForm
2020-11-11 16:07:38 -05:00
class TagBulkDeleteView(generic.BulkDeleteView):
queryset = Tag.objects.annotate(
items=count_related(TaggedItem, 'tag')
)
table = tables.TagTable
2018-06-27 16:02:34 -04:00
#
# Config contexts
#
2020-11-11 16:07:38 -05:00
class ConfigContextListView(generic.ObjectListView):
2018-06-27 16:02:34 -04:00
queryset = ConfigContext.objects.all()
2021-04-29 15:59:11 -04:00
filterset = filtersets.ConfigContextFilterSet
filterset_form = forms.ConfigContextFilterForm
table = tables.ConfigContextTable
actions = ('add', 'bulk_edit', 'bulk_delete')
2018-06-27 16:02:34 -04:00
2020-11-11 16:07:38 -05:00
class ConfigContextView(generic.ObjectView):
queryset = ConfigContext.objects.all()
2018-06-27 16:02:34 -04:00
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),
('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,
}
2018-06-27 16:02:34 -04:00
2020-11-11 16:07:38 -05:00
class ConfigContextEditView(generic.ObjectEditView):
queryset = ConfigContext.objects.all()
model_form = forms.ConfigContextForm
2018-06-27 16:02:34 -04:00
template_name = 'extras/configcontext_edit.html'
2020-11-11 16:07:38 -05:00
class ConfigContextBulkEditView(generic.BulkEditView):
queryset = ConfigContext.objects.all()
2021-04-29 15:59:11 -04:00
filterset = filtersets.ConfigContextFilterSet
table = tables.ConfigContextTable
form = forms.ConfigContextBulkEditForm
2020-11-11 16:07:38 -05:00
class ConfigContextDeleteView(generic.ObjectDeleteView):
queryset = ConfigContext.objects.all()
2018-06-27 16:02:34 -04:00
2020-11-11 16:07:38 -05:00
class ConfigContextBulkDeleteView(generic.BulkDeleteView):
2018-06-27 16:02:34 -04:00
queryset = ConfigContext.objects.all()
table = tables.ConfigContextTable
2018-06-27 16:02:34 -04:00
2020-11-11 16:07:38 -05:00
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,
'active_tab': 'config-context',
}
2018-06-14 16:15:14 -04:00
#
# Change logging
#
2020-11-11 16:07:38 -05:00
class ObjectChangeListView(generic.ObjectListView):
queryset = ObjectChange.objects.all()
2021-04-29 15:59:11 -04:00
filterset = filtersets.ObjectChangeFilterSet
filterset_form = forms.ObjectChangeFilterForm
table = tables.ObjectChangeTable
2018-06-20 13:52:54 -04:00
template_name = 'extras/objectchange_list.html'
actions = ('export',)
2018-06-20 13:52:54 -04:00
2020-11-11 16:07:38 -05:00
class ObjectChangeView(generic.ObjectView):
queryset = ObjectChange.objects.all()
2018-06-20 13:52:54 -04:00
def get_extra_context(self, request, instance):
2020-06-01 11:43:49 -04:00
related_changes = ObjectChange.objects.restrict(request.user, 'view').filter(
request_id=instance.request_id
2020-06-01 11:43:49 -04:00
).exclude(
pk=instance.pk
2020-06-01 11:43:49 -04:00
)
related_changes_table = tables.ObjectChangeTable(
2018-06-20 13:52:54 -04:00
data=related_changes[:50],
orderable=False
)
2020-06-01 11:43:49 -04:00
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,
2018-06-20 13:52:54 -04:00
'related_changes_table': related_changes_table,
'related_changes_count': related_changes.count(),
'non_atomic_change': non_atomic_change
}
2018-06-20 13:52:54 -04:00
2017-09-21 16:32:05 -04:00
#
# Image attachments
#
2020-11-11 16:07:38 -05:00
class ImageAttachmentEditView(generic.ObjectEditView):
queryset = ImageAttachment.objects.all()
model_form = forms.ImageAttachmentForm
template_name = 'extras/imageattachment_edit.html'
2021-12-14 11:28:13 -05:00
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)
2020-11-11 16:07:38 -05:00
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)
2017-09-21 16:32:05 -04:00
#
# Journal entries
#
2021-03-16 15:57:23 -04:00
class JournalEntryListView(generic.ObjectListView):
queryset = JournalEntry.objects.all()
2021-04-29 15:59:11 -04:00
filterset = filtersets.JournalEntryFilterSet
2021-03-16 15:57:23 -04:00
filterset_form = forms.JournalEntryFilterForm
table = tables.JournalEntryTable
actions = ('export', 'bulk_edit', 'bulk_delete')
2021-03-16 15:57:23 -04:00
class JournalEntryView(generic.ObjectView):
queryset = JournalEntry.objects.all()
class JournalEntryEditView(generic.ObjectEditView):
queryset = JournalEntry.objects.all()
model_form = forms.JournalEntryForm
2021-12-14 11:28:13 -05:00
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):
2021-03-16 16:47:35 -04:00
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})
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.prefetch_related('created_by')
2021-04-29 15:59:11 -04:00
filterset = filtersets.JournalEntryFilterSet
table = tables.JournalEntryTable
form = forms.JournalEntryBulkEditForm
class JournalEntryBulkDeleteView(generic.BulkDeleteView):
queryset = JournalEntry.objects.prefetch_related('created_by')
2021-04-29 15:59:11 -04:00
filterset = filtersets.JournalEntryFilterSet
table = tables.JournalEntryTable
2017-09-21 16:32:05 -04:00
#
# Reports
#
class ReportListView(ContentTypePermissionRequiredMixin, View):
2017-09-21 16:32:05 -04:00
"""
Retrieve all of the available reports from disk and the recorded JobResult (if any) for each.
2017-09-21 16:32:05 -04:00
"""
def get_required_permission(self):
return 'extras.view_report'
2017-09-21 16:32:05 -04:00
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
).defer('data')
}
ret = []
for module, report_list in reports:
module_reports = []
2017-09-26 16:36:43 -04:00
for report in report_list:
report.result = results.get(report.full_name, None)
module_reports.append(report)
ret.append((module, module_reports))
2017-09-21 16:32:05 -04:00
return render(request, 'extras/report_list.html', {
'reports': ret,
})
2020-08-10 15:56:55 -04:00
class ReportView(ContentTypePermissionRequiredMixin, View):
"""
Display a single Report and its associated JobResult (if any).
"""
def get_required_permission(self):
return 'extras.view_report'
2020-07-03 11:55:04 -04:00
def get(self, request, module, name):
2020-08-10 15:56:55 -04:00
report = get_report(module, name)
if report is None:
raise Http404
report_content_type = ContentType.objects.get(app_label='extras', model='report')
2020-07-03 11:55:04 -04:00
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,
2017-09-28 12:51:10 -04:00
'run_form': ConfirmationForm(),
2017-09-21 16:32:05 -04:00
})
2017-09-28 12:51:10 -04:00
2020-07-03 11:55:04 -04:00
def post(self, request, module, name):
2017-09-28 12:51:10 -04:00
2020-07-03 11:55:04 -04:00
# Permissions check
if not request.user.has_perm('extras.run_report'):
return HttpResponseForbidden()
2017-09-28 12:51:10 -04:00
2020-08-10 15:56:55 -04:00
report = get_report(module, name)
if report is None:
raise Http404
# 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.")
2020-08-10 15:56:55 -04:00
return render(request, 'extras/report.html', {
'report': report,
})
2020-08-10 15:56:55 -04:00
# Run the Report. A new JobResult is created.
report_content_type = ContentType.objects.get(app_label='extras', model='report')
job_result = JobResult.enqueue_job(
run_report,
report.full_name,
report_content_type,
request.user
)
2020-07-03 11:55:04 -04:00
2020-08-10 15:56:55 -04:00
return redirect('extras:report_result', job_result_pk=job_result.pk)
2019-08-09 12:33:33 -04:00
2020-07-03 11:55:04 -04:00
2020-08-10 15:56:55 -04:00
class ReportResultView(ContentTypePermissionRequiredMixin, View):
"""
Display a JobResult pertaining to the execution of a Report.
"""
2020-07-03 11:55:04 -04:00
def get_required_permission(self):
return 'extras.view_report'
2020-07-03 11:55:04 -04:00
def get(self, request, job_result_pk):
report_content_type = ContentType.objects.get(app_label='extras', model='report')
2021-12-20 20:52:29 -05:00
result = get_object_or_404(JobResult.objects.all(), pk=job_result_pk, obj_type=report_content_type)
2020-07-03 11:55:04 -04:00
2020-08-10 15:56:55 -04:00
# Retrieve the Report and attach the JobResult to it
2021-12-20 20:52:29 -05:00
module, report_name = result.name.split('.')
2020-08-10 15:56:55 -04:00
report = get_report(module, report_name)
2021-12-20 20:52:29 -05:00
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:
response.status_code = 286
return response
2020-07-03 11:55:04 -04:00
return render(request, 'extras/report_result.html', {
'report': report,
2021-12-20 20:52:29 -05:00
'result': result,
2020-07-03 11:55:04 -04:00
})
2019-08-09 12:33:33 -04:00
2020-07-06 10:44:36 -04:00
2019-08-09 12:33:33 -04:00
#
# Scripts
#
class GetScriptMixin:
2020-07-03 11:55:04 -04:00
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'
2019-08-09 12:33:33 -04:00
def get(self, request):
2020-07-06 01:58:28 -04:00
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
).defer('data')
}
for _scripts in scripts.values():
for script in _scripts.values():
script.result = results.get(script.full_name)
2019-08-09 12:33:33 -04:00
return render(request, 'extras/script_list.html', {
2020-07-06 01:58:28 -04:00
'scripts': scripts,
2019-08-09 12:33:33 -04:00
})
class ScriptView(ContentTypePermissionRequiredMixin, GetScriptMixin, View):
def get_required_permission(self):
return 'extras.view_script'
2019-08-09 12:33:33 -04:00
def get(self, request, module, name):
2020-07-03 11:55:04 -04:00
script = self._get_script(name, module)
form = script.as_form(initial=normalize_querydict(request.GET))
2019-08-09 12:33:33 -04:00
# Look for a pending JobResult (use the latest one by creation timestamp)
script_content_type = ContentType.objects.get(app_label='extras', model='script')
script.result = JobResult.objects.filter(
obj_type=script_content_type,
name=script.full_name,
2020-07-03 11:55:04 -04:00
).exclude(
status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
).first()
2019-08-09 12:33:33 -04:00
return render(request, 'extras/script.html', {
'module': module,
'script': script,
'form': form,
})
def post(self, request, module, name):
2019-08-12 11:39:36 -04:00
# Permissions check
if not request.user.has_perm('extras.run_script'):
return HttpResponseForbidden()
2020-07-03 11:55:04 -04:00
script = self._get_script(name, module)
2019-08-16 15:27:58 -04:00
form = script.as_form(request.POST, request.FILES)
2019-08-09 12:33:33 -04:00
# 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():
2019-08-12 14:28:06 -04:00
commit = form.cleaned_data.pop('_commit')
script_content_type = ContentType.objects.get(app_label='extras', model='script')
job_result = JobResult.enqueue_job(
run_script,
script.full_name,
script_content_type,
request.user,
data=form.cleaned_data,
request=copy_safe_request(request),
commit=commit
)
2019-08-09 12:33:33 -04:00
2020-07-03 11:55:04 -04:00
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'
2020-07-03 11:55:04 -04:00
def get(self, request, job_result_pk):
result = get_object_or_404(JobResult.objects.all(), pk=job_result_pk)
2020-07-03 11:55:04 -04:00
script_content_type = ContentType.objects.get(app_label='extras', model='script')
if result.obj_type != script_content_type:
raise Http404
2020-07-03 11:55:04 -04:00
script = self._get_script(result.name)
2021-12-20 20:52:29 -05:00
# If this is an HTMX request, return only the result HTML
2021-12-20 20:45:32 -05:00
if is_htmx(request):
response = render(request, 'extras/htmx/script_result.html', {
'script': script,
'result': result,
})
if result.completed:
response.status_code = 286
return response
return render(request, 'extras/script_result.html', {
2019-08-09 12:33:33 -04:00
'script': script,
'result': result,
2020-07-03 11:55:04 -04:00
'class_name': script.__class__.__name__
2019-08-09 12:33:33 -04:00
})