views.py 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766
  1. from django.contrib import messages
  2. from django.contrib.contenttypes.models import ContentType
  3. from django.db.models import Count, Q
  4. from django.http import Http404, HttpResponseForbidden
  5. from django.shortcuts import get_object_or_404, redirect, render
  6. from django.urls import reverse
  7. from django.views.generic import View
  8. from django_rq.queues import get_connection
  9. from rq import Worker
  10. from netbox.views import generic
  11. from utilities.forms import ConfirmationForm
  12. from utilities.htmx import is_htmx
  13. from utilities.utils import copy_safe_request, count_related, get_viewname, normalize_querydict, shallow_compare_dict
  14. from utilities.views import ContentTypePermissionRequiredMixin
  15. from . import filtersets, forms, tables
  16. from .choices import JobResultStatusChoices
  17. from .models import *
  18. from .reports import get_report, get_reports, run_report
  19. from .scripts import get_scripts, run_script
  20. #
  21. # Custom fields
  22. #
  23. class CustomFieldListView(generic.ObjectListView):
  24. queryset = CustomField.objects.all()
  25. filterset = filtersets.CustomFieldFilterSet
  26. filterset_form = forms.CustomFieldFilterForm
  27. table = tables.CustomFieldTable
  28. class CustomFieldView(generic.ObjectView):
  29. queryset = CustomField.objects.all()
  30. class CustomFieldEditView(generic.ObjectEditView):
  31. queryset = CustomField.objects.all()
  32. form = forms.CustomFieldForm
  33. class CustomFieldDeleteView(generic.ObjectDeleteView):
  34. queryset = CustomField.objects.all()
  35. class CustomFieldBulkImportView(generic.BulkImportView):
  36. queryset = CustomField.objects.all()
  37. model_form = forms.CustomFieldCSVForm
  38. table = tables.CustomFieldTable
  39. class CustomFieldBulkEditView(generic.BulkEditView):
  40. queryset = CustomField.objects.all()
  41. filterset = filtersets.CustomFieldFilterSet
  42. table = tables.CustomFieldTable
  43. form = forms.CustomFieldBulkEditForm
  44. class CustomFieldBulkDeleteView(generic.BulkDeleteView):
  45. queryset = CustomField.objects.all()
  46. filterset = filtersets.CustomFieldFilterSet
  47. table = tables.CustomFieldTable
  48. #
  49. # Custom links
  50. #
  51. class CustomLinkListView(generic.ObjectListView):
  52. queryset = CustomLink.objects.all()
  53. filterset = filtersets.CustomLinkFilterSet
  54. filterset_form = forms.CustomLinkFilterForm
  55. table = tables.CustomLinkTable
  56. class CustomLinkView(generic.ObjectView):
  57. queryset = CustomLink.objects.all()
  58. class CustomLinkEditView(generic.ObjectEditView):
  59. queryset = CustomLink.objects.all()
  60. form = forms.CustomLinkForm
  61. class CustomLinkDeleteView(generic.ObjectDeleteView):
  62. queryset = CustomLink.objects.all()
  63. class CustomLinkBulkImportView(generic.BulkImportView):
  64. queryset = CustomLink.objects.all()
  65. model_form = forms.CustomLinkCSVForm
  66. table = tables.CustomLinkTable
  67. class CustomLinkBulkEditView(generic.BulkEditView):
  68. queryset = CustomLink.objects.all()
  69. filterset = filtersets.CustomLinkFilterSet
  70. table = tables.CustomLinkTable
  71. form = forms.CustomLinkBulkEditForm
  72. class CustomLinkBulkDeleteView(generic.BulkDeleteView):
  73. queryset = CustomLink.objects.all()
  74. filterset = filtersets.CustomLinkFilterSet
  75. table = tables.CustomLinkTable
  76. #
  77. # Export templates
  78. #
  79. class ExportTemplateListView(generic.ObjectListView):
  80. queryset = ExportTemplate.objects.all()
  81. filterset = filtersets.ExportTemplateFilterSet
  82. filterset_form = forms.ExportTemplateFilterForm
  83. table = tables.ExportTemplateTable
  84. class ExportTemplateView(generic.ObjectView):
  85. queryset = ExportTemplate.objects.all()
  86. class ExportTemplateEditView(generic.ObjectEditView):
  87. queryset = ExportTemplate.objects.all()
  88. form = forms.ExportTemplateForm
  89. class ExportTemplateDeleteView(generic.ObjectDeleteView):
  90. queryset = ExportTemplate.objects.all()
  91. class ExportTemplateBulkImportView(generic.BulkImportView):
  92. queryset = ExportTemplate.objects.all()
  93. model_form = forms.ExportTemplateCSVForm
  94. table = tables.ExportTemplateTable
  95. class ExportTemplateBulkEditView(generic.BulkEditView):
  96. queryset = ExportTemplate.objects.all()
  97. filterset = filtersets.ExportTemplateFilterSet
  98. table = tables.ExportTemplateTable
  99. form = forms.ExportTemplateBulkEditForm
  100. class ExportTemplateBulkDeleteView(generic.BulkDeleteView):
  101. queryset = ExportTemplate.objects.all()
  102. filterset = filtersets.ExportTemplateFilterSet
  103. table = tables.ExportTemplateTable
  104. #
  105. # Webhooks
  106. #
  107. class WebhookListView(generic.ObjectListView):
  108. queryset = Webhook.objects.all()
  109. filterset = filtersets.WebhookFilterSet
  110. filterset_form = forms.WebhookFilterForm
  111. table = tables.WebhookTable
  112. class WebhookView(generic.ObjectView):
  113. queryset = Webhook.objects.all()
  114. class WebhookEditView(generic.ObjectEditView):
  115. queryset = Webhook.objects.all()
  116. form = forms.WebhookForm
  117. class WebhookDeleteView(generic.ObjectDeleteView):
  118. queryset = Webhook.objects.all()
  119. class WebhookBulkImportView(generic.BulkImportView):
  120. queryset = Webhook.objects.all()
  121. model_form = forms.WebhookCSVForm
  122. table = tables.WebhookTable
  123. class WebhookBulkEditView(generic.BulkEditView):
  124. queryset = Webhook.objects.all()
  125. filterset = filtersets.WebhookFilterSet
  126. table = tables.WebhookTable
  127. form = forms.WebhookBulkEditForm
  128. class WebhookBulkDeleteView(generic.BulkDeleteView):
  129. queryset = Webhook.objects.all()
  130. filterset = filtersets.WebhookFilterSet
  131. table = tables.WebhookTable
  132. #
  133. # Tags
  134. #
  135. class TagListView(generic.ObjectListView):
  136. queryset = Tag.objects.annotate(
  137. items=count_related(TaggedItem, 'tag')
  138. )
  139. filterset = filtersets.TagFilterSet
  140. filterset_form = forms.TagFilterForm
  141. table = tables.TagTable
  142. class TagView(generic.ObjectView):
  143. queryset = Tag.objects.all()
  144. def get_extra_context(self, request, instance):
  145. tagged_items = TaggedItem.objects.filter(tag=instance)
  146. taggeditem_table = tables.TaggedItemTable(
  147. data=tagged_items,
  148. orderable=False
  149. )
  150. taggeditem_table.configure(request)
  151. object_types = [
  152. {
  153. 'content_type': ContentType.objects.get(pk=ti['content_type']),
  154. 'item_count': ti['item_count']
  155. } for ti in tagged_items.values('content_type').annotate(item_count=Count('pk'))
  156. ]
  157. return {
  158. 'taggeditem_table': taggeditem_table,
  159. 'tagged_item_count': tagged_items.count(),
  160. 'object_types': object_types,
  161. }
  162. class TagEditView(generic.ObjectEditView):
  163. queryset = Tag.objects.all()
  164. form = forms.TagForm
  165. class TagDeleteView(generic.ObjectDeleteView):
  166. queryset = Tag.objects.all()
  167. class TagBulkImportView(generic.BulkImportView):
  168. queryset = Tag.objects.all()
  169. model_form = forms.TagCSVForm
  170. table = tables.TagTable
  171. class TagBulkEditView(generic.BulkEditView):
  172. queryset = Tag.objects.annotate(
  173. items=count_related(TaggedItem, 'tag')
  174. )
  175. table = tables.TagTable
  176. form = forms.TagBulkEditForm
  177. class TagBulkDeleteView(generic.BulkDeleteView):
  178. queryset = Tag.objects.annotate(
  179. items=count_related(TaggedItem, 'tag')
  180. )
  181. table = tables.TagTable
  182. #
  183. # Config contexts
  184. #
  185. class ConfigContextListView(generic.ObjectListView):
  186. queryset = ConfigContext.objects.all()
  187. filterset = filtersets.ConfigContextFilterSet
  188. filterset_form = forms.ConfigContextFilterForm
  189. table = tables.ConfigContextTable
  190. actions = ('add', 'bulk_edit', 'bulk_delete')
  191. class ConfigContextView(generic.ObjectView):
  192. queryset = ConfigContext.objects.all()
  193. def get_extra_context(self, request, instance):
  194. # Gather assigned objects for parsing in the template
  195. assigned_objects = (
  196. ('Regions', instance.regions.all),
  197. ('Site Groups', instance.site_groups.all),
  198. ('Sites', instance.sites.all),
  199. ('Locations', instance.locations.all),
  200. ('Device Types', instance.device_types.all),
  201. ('Roles', instance.roles.all),
  202. ('Platforms', instance.platforms.all),
  203. ('Cluster Types', instance.cluster_types.all),
  204. ('Cluster Groups', instance.cluster_groups.all),
  205. ('Clusters', instance.clusters.all),
  206. ('Tenant Groups', instance.tenant_groups.all),
  207. ('Tenants', instance.tenants.all),
  208. ('Tags', instance.tags.all),
  209. )
  210. # Determine user's preferred output format
  211. if request.GET.get('format') in ['json', 'yaml']:
  212. format = request.GET.get('format')
  213. if request.user.is_authenticated:
  214. request.user.config.set('data_format', format, commit=True)
  215. elif request.user.is_authenticated:
  216. format = request.user.config.get('data_format', 'json')
  217. else:
  218. format = 'json'
  219. return {
  220. 'assigned_objects': assigned_objects,
  221. 'format': format,
  222. }
  223. class ConfigContextEditView(generic.ObjectEditView):
  224. queryset = ConfigContext.objects.all()
  225. form = forms.ConfigContextForm
  226. class ConfigContextBulkEditView(generic.BulkEditView):
  227. queryset = ConfigContext.objects.all()
  228. filterset = filtersets.ConfigContextFilterSet
  229. table = tables.ConfigContextTable
  230. form = forms.ConfigContextBulkEditForm
  231. class ConfigContextDeleteView(generic.ObjectDeleteView):
  232. queryset = ConfigContext.objects.all()
  233. class ConfigContextBulkDeleteView(generic.BulkDeleteView):
  234. queryset = ConfigContext.objects.all()
  235. table = tables.ConfigContextTable
  236. class ObjectConfigContextView(generic.ObjectView):
  237. base_template = None
  238. template_name = 'extras/object_configcontext.html'
  239. def get_extra_context(self, request, instance):
  240. source_contexts = ConfigContext.objects.restrict(request.user, 'view').get_for_object(instance)
  241. # Determine user's preferred output format
  242. if request.GET.get('format') in ['json', 'yaml']:
  243. format = request.GET.get('format')
  244. if request.user.is_authenticated:
  245. request.user.config.set('data_format', format, commit=True)
  246. elif request.user.is_authenticated:
  247. format = request.user.config.get('data_format', 'json')
  248. else:
  249. format = 'json'
  250. return {
  251. 'rendered_context': instance.get_config_context(),
  252. 'source_contexts': source_contexts,
  253. 'format': format,
  254. 'base_template': self.base_template,
  255. 'active_tab': 'config-context',
  256. }
  257. #
  258. # Change logging
  259. #
  260. class ObjectChangeListView(generic.ObjectListView):
  261. queryset = ObjectChange.objects.all()
  262. filterset = filtersets.ObjectChangeFilterSet
  263. filterset_form = forms.ObjectChangeFilterForm
  264. table = tables.ObjectChangeTable
  265. template_name = 'extras/objectchange_list.html'
  266. actions = ('export',)
  267. class ObjectChangeView(generic.ObjectView):
  268. queryset = ObjectChange.objects.all()
  269. def get_extra_context(self, request, instance):
  270. related_changes = ObjectChange.objects.restrict(request.user, 'view').filter(
  271. request_id=instance.request_id
  272. ).exclude(
  273. pk=instance.pk
  274. )
  275. related_changes_table = tables.ObjectChangeTable(
  276. data=related_changes[:50],
  277. orderable=False
  278. )
  279. objectchanges = ObjectChange.objects.restrict(request.user, 'view').filter(
  280. changed_object_type=instance.changed_object_type,
  281. changed_object_id=instance.changed_object_id,
  282. )
  283. next_change = objectchanges.filter(time__gt=instance.time).order_by('time').first()
  284. prev_change = objectchanges.filter(time__lt=instance.time).order_by('-time').first()
  285. if not instance.prechange_data and instance.action in ['update', 'delete'] and prev_change:
  286. non_atomic_change = True
  287. prechange_data = prev_change.postchange_data
  288. else:
  289. non_atomic_change = False
  290. prechange_data = instance.prechange_data
  291. if prechange_data and instance.postchange_data:
  292. diff_added = shallow_compare_dict(
  293. prechange_data or dict(),
  294. instance.postchange_data or dict(),
  295. exclude=['last_updated'],
  296. )
  297. diff_removed = {
  298. x: prechange_data.get(x) for x in diff_added
  299. } if prechange_data else {}
  300. else:
  301. diff_added = None
  302. diff_removed = None
  303. return {
  304. 'diff_added': diff_added,
  305. 'diff_removed': diff_removed,
  306. 'next_change': next_change,
  307. 'prev_change': prev_change,
  308. 'related_changes_table': related_changes_table,
  309. 'related_changes_count': related_changes.count(),
  310. 'non_atomic_change': non_atomic_change
  311. }
  312. #
  313. # Image attachments
  314. #
  315. class ImageAttachmentEditView(generic.ObjectEditView):
  316. queryset = ImageAttachment.objects.all()
  317. form = forms.ImageAttachmentForm
  318. template_name = 'extras/imageattachment_edit.html'
  319. def alter_object(self, instance, request, args, kwargs):
  320. if not instance.pk:
  321. # Assign the parent object based on URL kwargs
  322. content_type = get_object_or_404(ContentType, pk=request.GET.get('content_type'))
  323. instance.parent = get_object_or_404(content_type.model_class(), pk=request.GET.get('object_id'))
  324. return instance
  325. def get_return_url(self, request, obj=None):
  326. return obj.parent.get_absolute_url() if obj else super().get_return_url(request)
  327. def get_extra_addanother_params(self, request):
  328. return {
  329. 'content_type': request.GET.get('content_type'),
  330. 'object_id': request.GET.get('object_id'),
  331. }
  332. class ImageAttachmentDeleteView(generic.ObjectDeleteView):
  333. queryset = ImageAttachment.objects.all()
  334. def get_return_url(self, request, obj=None):
  335. return obj.parent.get_absolute_url() if obj else super().get_return_url(request)
  336. #
  337. # Journal entries
  338. #
  339. class JournalEntryListView(generic.ObjectListView):
  340. queryset = JournalEntry.objects.all()
  341. filterset = filtersets.JournalEntryFilterSet
  342. filterset_form = forms.JournalEntryFilterForm
  343. table = tables.JournalEntryTable
  344. actions = ('export', 'bulk_edit', 'bulk_delete')
  345. class JournalEntryView(generic.ObjectView):
  346. queryset = JournalEntry.objects.all()
  347. class JournalEntryEditView(generic.ObjectEditView):
  348. queryset = JournalEntry.objects.all()
  349. form = forms.JournalEntryForm
  350. def alter_object(self, obj, request, args, kwargs):
  351. if not obj.pk:
  352. obj.created_by = request.user
  353. return obj
  354. def get_return_url(self, request, instance):
  355. if not instance.assigned_object:
  356. return reverse('extras:journalentry_list')
  357. obj = instance.assigned_object
  358. viewname = get_viewname(obj, 'journal')
  359. return reverse(viewname, kwargs={'pk': obj.pk})
  360. class JournalEntryDeleteView(generic.ObjectDeleteView):
  361. queryset = JournalEntry.objects.all()
  362. def get_return_url(self, request, instance):
  363. obj = instance.assigned_object
  364. viewname = get_viewname(obj, 'journal')
  365. return reverse(viewname, kwargs={'pk': obj.pk})
  366. class JournalEntryBulkEditView(generic.BulkEditView):
  367. queryset = JournalEntry.objects.all()
  368. filterset = filtersets.JournalEntryFilterSet
  369. table = tables.JournalEntryTable
  370. form = forms.JournalEntryBulkEditForm
  371. class JournalEntryBulkDeleteView(generic.BulkDeleteView):
  372. queryset = JournalEntry.objects.all()
  373. filterset = filtersets.JournalEntryFilterSet
  374. table = tables.JournalEntryTable
  375. #
  376. # Reports
  377. #
  378. class ReportListView(ContentTypePermissionRequiredMixin, View):
  379. """
  380. Retrieve all of the available reports from disk and the recorded JobResult (if any) for each.
  381. """
  382. def get_required_permission(self):
  383. return 'extras.view_report'
  384. def get(self, request):
  385. reports = get_reports()
  386. report_content_type = ContentType.objects.get(app_label='extras', model='report')
  387. results = {
  388. r.name: r
  389. for r in JobResult.objects.filter(
  390. obj_type=report_content_type,
  391. status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
  392. ).order_by('name', '-created').distinct('name').defer('data')
  393. }
  394. ret = []
  395. for module, report_list in reports:
  396. module_reports = []
  397. for report in report_list:
  398. report.result = results.get(report.full_name, None)
  399. module_reports.append(report)
  400. ret.append((module, module_reports))
  401. return render(request, 'extras/report_list.html', {
  402. 'reports': ret,
  403. })
  404. class ReportView(ContentTypePermissionRequiredMixin, View):
  405. """
  406. Display a single Report and its associated JobResult (if any).
  407. """
  408. def get_required_permission(self):
  409. return 'extras.view_report'
  410. def get(self, request, module, name):
  411. report = get_report(module, name)
  412. if report is None:
  413. raise Http404
  414. report_content_type = ContentType.objects.get(app_label='extras', model='report')
  415. report.result = JobResult.objects.filter(
  416. obj_type=report_content_type,
  417. name=report.full_name,
  418. status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
  419. ).first()
  420. return render(request, 'extras/report.html', {
  421. 'report': report,
  422. 'run_form': ConfirmationForm(),
  423. })
  424. def post(self, request, module, name):
  425. # Permissions check
  426. if not request.user.has_perm('extras.run_report'):
  427. return HttpResponseForbidden()
  428. report = get_report(module, name)
  429. if report is None:
  430. raise Http404
  431. # Allow execution only if RQ worker process is running
  432. if not Worker.count(get_connection('default')):
  433. messages.error(request, "Unable to run report: RQ worker process not running.")
  434. return render(request, 'extras/report.html', {
  435. 'report': report,
  436. })
  437. # Run the Report. A new JobResult is created.
  438. report_content_type = ContentType.objects.get(app_label='extras', model='report')
  439. job_result = JobResult.enqueue_job(
  440. run_report,
  441. report.full_name,
  442. report_content_type,
  443. request.user,
  444. job_timeout=report.job_timeout
  445. )
  446. return redirect('extras:report_result', job_result_pk=job_result.pk)
  447. class ReportResultView(ContentTypePermissionRequiredMixin, View):
  448. """
  449. Display a JobResult pertaining to the execution of a Report.
  450. """
  451. def get_required_permission(self):
  452. return 'extras.view_report'
  453. def get(self, request, job_result_pk):
  454. report_content_type = ContentType.objects.get(app_label='extras', model='report')
  455. result = get_object_or_404(JobResult.objects.all(), pk=job_result_pk, obj_type=report_content_type)
  456. # Retrieve the Report and attach the JobResult to it
  457. module, report_name = result.name.split('.')
  458. report = get_report(module, report_name)
  459. report.result = result
  460. # If this is an HTMX request, return only the result HTML
  461. if is_htmx(request):
  462. response = render(request, 'extras/htmx/report_result.html', {
  463. 'report': report,
  464. 'result': result,
  465. })
  466. if result.completed:
  467. response.status_code = 286
  468. return response
  469. return render(request, 'extras/report_result.html', {
  470. 'report': report,
  471. 'result': result,
  472. })
  473. #
  474. # Scripts
  475. #
  476. class GetScriptMixin:
  477. def _get_script(self, name, module=None):
  478. if module is None:
  479. module, name = name.split('.', 1)
  480. scripts = get_scripts()
  481. try:
  482. return scripts[module][name]()
  483. except KeyError:
  484. raise Http404
  485. class ScriptListView(ContentTypePermissionRequiredMixin, View):
  486. def get_required_permission(self):
  487. return 'extras.view_script'
  488. def get(self, request):
  489. scripts = get_scripts(use_names=True)
  490. script_content_type = ContentType.objects.get(app_label='extras', model='script')
  491. results = {
  492. r.name: r
  493. for r in JobResult.objects.filter(
  494. obj_type=script_content_type,
  495. status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
  496. ).order_by('name', '-created').distinct('name').defer('data')
  497. }
  498. for _scripts in scripts.values():
  499. for script in _scripts.values():
  500. script.result = results.get(script.full_name)
  501. return render(request, 'extras/script_list.html', {
  502. 'scripts': scripts,
  503. })
  504. class ScriptView(ContentTypePermissionRequiredMixin, GetScriptMixin, View):
  505. def get_required_permission(self):
  506. return 'extras.view_script'
  507. def get(self, request, module, name):
  508. script = self._get_script(name, module)
  509. form = script.as_form(initial=normalize_querydict(request.GET))
  510. # Look for a pending JobResult (use the latest one by creation timestamp)
  511. script_content_type = ContentType.objects.get(app_label='extras', model='script')
  512. script.result = JobResult.objects.filter(
  513. obj_type=script_content_type,
  514. name=script.full_name,
  515. ).exclude(
  516. status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
  517. ).first()
  518. return render(request, 'extras/script.html', {
  519. 'module': module,
  520. 'script': script,
  521. 'form': form,
  522. })
  523. def post(self, request, module, name):
  524. # Permissions check
  525. if not request.user.has_perm('extras.run_script'):
  526. return HttpResponseForbidden()
  527. script = self._get_script(name, module)
  528. form = script.as_form(request.POST, request.FILES)
  529. # Allow execution only if RQ worker process is running
  530. if not Worker.count(get_connection('default')):
  531. messages.error(request, "Unable to run script: RQ worker process not running.")
  532. elif form.is_valid():
  533. commit = form.cleaned_data.pop('_commit')
  534. script_content_type = ContentType.objects.get(app_label='extras', model='script')
  535. job_result = JobResult.enqueue_job(
  536. run_script,
  537. script.full_name,
  538. script_content_type,
  539. request.user,
  540. data=form.cleaned_data,
  541. request=copy_safe_request(request),
  542. commit=commit,
  543. job_timeout=script.job_timeout,
  544. )
  545. return redirect('extras:script_result', job_result_pk=job_result.pk)
  546. return render(request, 'extras/script.html', {
  547. 'module': module,
  548. 'script': script,
  549. 'form': form,
  550. })
  551. class ScriptResultView(ContentTypePermissionRequiredMixin, GetScriptMixin, View):
  552. def get_required_permission(self):
  553. return 'extras.view_script'
  554. def get(self, request, job_result_pk):
  555. result = get_object_or_404(JobResult.objects.all(), pk=job_result_pk)
  556. script_content_type = ContentType.objects.get(app_label='extras', model='script')
  557. if result.obj_type != script_content_type:
  558. raise Http404
  559. script = self._get_script(result.name)
  560. # If this is an HTMX request, return only the result HTML
  561. if is_htmx(request):
  562. response = render(request, 'extras/htmx/script_result.html', {
  563. 'script': script,
  564. 'result': result,
  565. })
  566. if result.completed:
  567. response.status_code = 286
  568. return response
  569. return render(request, 'extras/script_result.html', {
  570. 'script': script,
  571. 'result': result,
  572. 'class_name': script.__class__.__name__
  573. })