views.py 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850
  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.tables import paginate_table
  14. from utilities.utils import copy_safe_request, count_related, normalize_querydict, shallow_compare_dict
  15. from utilities.views import ContentTypePermissionRequiredMixin
  16. from . import filtersets, forms, tables
  17. from .choices import JobResultStatusChoices
  18. from .models import *
  19. from .reports import get_report, get_reports, run_report
  20. from .scripts import get_scripts, run_script
  21. #
  22. # Custom fields
  23. #
  24. class CustomFieldListView(generic.ObjectListView):
  25. queryset = CustomField.objects.all()
  26. filterset = filtersets.CustomFieldFilterSet
  27. filterset_form = forms.CustomFieldFilterForm
  28. table = tables.CustomFieldTable
  29. class CustomFieldView(generic.ObjectView):
  30. queryset = CustomField.objects.all()
  31. class CustomFieldEditView(generic.ObjectEditView):
  32. queryset = CustomField.objects.all()
  33. model_form = forms.CustomFieldForm
  34. class CustomFieldDeleteView(generic.ObjectDeleteView):
  35. queryset = CustomField.objects.all()
  36. class CustomFieldBulkImportView(generic.BulkImportView):
  37. queryset = CustomField.objects.all()
  38. model_form = forms.CustomFieldCSVForm
  39. table = tables.CustomFieldTable
  40. class CustomFieldBulkEditView(generic.BulkEditView):
  41. queryset = CustomField.objects.all()
  42. filterset = filtersets.CustomFieldFilterSet
  43. table = tables.CustomFieldTable
  44. form = forms.CustomFieldBulkEditForm
  45. class CustomFieldBulkDeleteView(generic.BulkDeleteView):
  46. queryset = CustomField.objects.all()
  47. filterset = filtersets.CustomFieldFilterSet
  48. table = tables.CustomFieldTable
  49. #
  50. # Custom links
  51. #
  52. class CustomLinkListView(generic.ObjectListView):
  53. queryset = CustomLink.objects.all()
  54. filterset = filtersets.CustomLinkFilterSet
  55. filterset_form = forms.CustomLinkFilterForm
  56. table = tables.CustomLinkTable
  57. class CustomLinkView(generic.ObjectView):
  58. queryset = CustomLink.objects.all()
  59. class CustomLinkEditView(generic.ObjectEditView):
  60. queryset = CustomLink.objects.all()
  61. model_form = forms.CustomLinkForm
  62. class CustomLinkDeleteView(generic.ObjectDeleteView):
  63. queryset = CustomLink.objects.all()
  64. class CustomLinkBulkImportView(generic.BulkImportView):
  65. queryset = CustomLink.objects.all()
  66. model_form = forms.CustomLinkCSVForm
  67. table = tables.CustomLinkTable
  68. class CustomLinkBulkEditView(generic.BulkEditView):
  69. queryset = CustomLink.objects.all()
  70. filterset = filtersets.CustomLinkFilterSet
  71. table = tables.CustomLinkTable
  72. form = forms.CustomLinkBulkEditForm
  73. class CustomLinkBulkDeleteView(generic.BulkDeleteView):
  74. queryset = CustomLink.objects.all()
  75. filterset = filtersets.CustomLinkFilterSet
  76. table = tables.CustomLinkTable
  77. #
  78. # Export templates
  79. #
  80. class ExportTemplateListView(generic.ObjectListView):
  81. queryset = ExportTemplate.objects.all()
  82. filterset = filtersets.ExportTemplateFilterSet
  83. filterset_form = forms.ExportTemplateFilterForm
  84. table = tables.ExportTemplateTable
  85. class ExportTemplateView(generic.ObjectView):
  86. queryset = ExportTemplate.objects.all()
  87. class ExportTemplateEditView(generic.ObjectEditView):
  88. queryset = ExportTemplate.objects.all()
  89. model_form = forms.ExportTemplateForm
  90. class ExportTemplateDeleteView(generic.ObjectDeleteView):
  91. queryset = ExportTemplate.objects.all()
  92. class ExportTemplateBulkImportView(generic.BulkImportView):
  93. queryset = ExportTemplate.objects.all()
  94. model_form = forms.ExportTemplateCSVForm
  95. table = tables.ExportTemplateTable
  96. class ExportTemplateBulkEditView(generic.BulkEditView):
  97. queryset = ExportTemplate.objects.all()
  98. filterset = filtersets.ExportTemplateFilterSet
  99. table = tables.ExportTemplateTable
  100. form = forms.ExportTemplateBulkEditForm
  101. class ExportTemplateBulkDeleteView(generic.BulkDeleteView):
  102. queryset = ExportTemplate.objects.all()
  103. filterset = filtersets.ExportTemplateFilterSet
  104. table = tables.ExportTemplateTable
  105. #
  106. # Webhooks
  107. #
  108. class WebhookListView(generic.ObjectListView):
  109. queryset = Webhook.objects.all()
  110. filterset = filtersets.WebhookFilterSet
  111. filterset_form = forms.WebhookFilterForm
  112. table = tables.WebhookTable
  113. class WebhookView(generic.ObjectView):
  114. queryset = Webhook.objects.all()
  115. class WebhookEditView(generic.ObjectEditView):
  116. queryset = Webhook.objects.all()
  117. model_form = forms.WebhookForm
  118. class WebhookDeleteView(generic.ObjectDeleteView):
  119. queryset = Webhook.objects.all()
  120. class WebhookBulkImportView(generic.BulkImportView):
  121. queryset = Webhook.objects.all()
  122. model_form = forms.WebhookCSVForm
  123. table = tables.WebhookTable
  124. class WebhookBulkEditView(generic.BulkEditView):
  125. queryset = Webhook.objects.all()
  126. filterset = filtersets.WebhookFilterSet
  127. table = tables.WebhookTable
  128. form = forms.WebhookBulkEditForm
  129. class WebhookBulkDeleteView(generic.BulkDeleteView):
  130. queryset = Webhook.objects.all()
  131. filterset = filtersets.WebhookFilterSet
  132. table = tables.WebhookTable
  133. #
  134. # Tags
  135. #
  136. class TagListView(generic.ObjectListView):
  137. queryset = Tag.objects.annotate(
  138. items=count_related(TaggedItem, 'tag')
  139. )
  140. filterset = filtersets.TagFilterSet
  141. filterset_form = forms.TagFilterForm
  142. table = tables.TagTable
  143. class TagView(generic.ObjectView):
  144. queryset = Tag.objects.all()
  145. def get_extra_context(self, request, instance):
  146. tagged_items = TaggedItem.objects.filter(tag=instance)
  147. taggeditem_table = tables.TaggedItemTable(
  148. data=tagged_items,
  149. orderable=False
  150. )
  151. paginate_table(taggeditem_table, request)
  152. object_types = [
  153. {
  154. 'content_type': ContentType.objects.get(pk=ti['content_type']),
  155. 'item_count': ti['item_count']
  156. } for ti in tagged_items.values('content_type').annotate(item_count=Count('pk'))
  157. ]
  158. return {
  159. 'taggeditem_table': taggeditem_table,
  160. 'tagged_item_count': tagged_items.count(),
  161. 'object_types': object_types,
  162. }
  163. class TagEditView(generic.ObjectEditView):
  164. queryset = Tag.objects.all()
  165. model_form = forms.TagForm
  166. class TagDeleteView(generic.ObjectDeleteView):
  167. queryset = Tag.objects.all()
  168. class TagBulkImportView(generic.BulkImportView):
  169. queryset = Tag.objects.all()
  170. model_form = forms.TagCSVForm
  171. table = tables.TagTable
  172. class TagBulkEditView(generic.BulkEditView):
  173. queryset = Tag.objects.annotate(
  174. items=count_related(TaggedItem, 'tag')
  175. )
  176. table = tables.TagTable
  177. form = forms.TagBulkEditForm
  178. class TagBulkDeleteView(generic.BulkDeleteView):
  179. queryset = Tag.objects.annotate(
  180. items=count_related(TaggedItem, 'tag')
  181. )
  182. table = tables.TagTable
  183. #
  184. # Config contexts
  185. #
  186. class ConfigContextListView(generic.ObjectListView):
  187. queryset = ConfigContext.objects.all()
  188. filterset = filtersets.ConfigContextFilterSet
  189. filterset_form = forms.ConfigContextFilterForm
  190. table = tables.ConfigContextTable
  191. action_buttons = ('add',)
  192. class ConfigContextView(generic.ObjectView):
  193. queryset = ConfigContext.objects.all()
  194. def get_extra_context(self, request, instance):
  195. # Gather assigned objects for parsing in the template
  196. assigned_objects = (
  197. ('Regions', instance.regions.all),
  198. ('Site Groups', instance.site_groups.all),
  199. ('Sites', instance.sites.all),
  200. ('Device Types', instance.device_types.all),
  201. ('Roles', instance.roles.all),
  202. ('Platforms', instance.platforms.all),
  203. ('Cluster Groups', instance.cluster_groups.all),
  204. ('Clusters', instance.clusters.all),
  205. ('Tenant Groups', instance.tenant_groups.all),
  206. ('Tenants', instance.tenants.all),
  207. ('Tags', instance.tags.all),
  208. )
  209. # Determine user's preferred output format
  210. if request.GET.get('format') in ['json', 'yaml']:
  211. format = request.GET.get('format')
  212. if request.user.is_authenticated:
  213. request.user.config.set('extras.configcontext.format', format, commit=True)
  214. elif request.user.is_authenticated:
  215. format = request.user.config.get('extras.configcontext.format', 'json')
  216. else:
  217. format = 'json'
  218. return {
  219. 'assigned_objects': assigned_objects,
  220. 'format': format,
  221. }
  222. class ConfigContextEditView(generic.ObjectEditView):
  223. queryset = ConfigContext.objects.all()
  224. model_form = forms.ConfigContextForm
  225. template_name = 'extras/configcontext_edit.html'
  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('extras.configcontext.format', format, commit=True)
  246. elif request.user.is_authenticated:
  247. format = request.user.config.get('extras.configcontext.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. action_buttons = ('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. class ObjectChangeLogView(View):
  313. """
  314. Present a history of changes made to a particular object.
  315. base_template: The name of the template to extend. If not provided, "<app>/<model>.html" will be used.
  316. """
  317. base_template = None
  318. def get(self, request, model, **kwargs):
  319. # Handle QuerySet restriction of parent object if needed
  320. if hasattr(model.objects, 'restrict'):
  321. obj = get_object_or_404(model.objects.restrict(request.user, 'view'), **kwargs)
  322. else:
  323. obj = get_object_or_404(model, **kwargs)
  324. # Gather all changes for this object (and its related objects)
  325. content_type = ContentType.objects.get_for_model(model)
  326. objectchanges = ObjectChange.objects.restrict(request.user, 'view').prefetch_related(
  327. 'user', 'changed_object_type'
  328. ).filter(
  329. Q(changed_object_type=content_type, changed_object_id=obj.pk) |
  330. Q(related_object_type=content_type, related_object_id=obj.pk)
  331. )
  332. objectchanges_table = tables.ObjectChangeTable(
  333. data=objectchanges,
  334. orderable=False,
  335. user=request.user
  336. )
  337. paginate_table(objectchanges_table, request)
  338. # Default to using "<app>/<model>.html" as the template, if it exists. Otherwise,
  339. # fall back to using base.html.
  340. if self.base_template is None:
  341. self.base_template = f"{model._meta.app_label}/{model._meta.model_name}.html"
  342. return render(request, 'extras/object_changelog.html', {
  343. 'object': obj,
  344. 'table': objectchanges_table,
  345. 'base_template': self.base_template,
  346. 'active_tab': 'changelog',
  347. })
  348. #
  349. # Image attachments
  350. #
  351. class ImageAttachmentEditView(generic.ObjectEditView):
  352. queryset = ImageAttachment.objects.all()
  353. model_form = forms.ImageAttachmentForm
  354. template_name = 'extras/imageattachment_edit.html'
  355. def alter_obj(self, instance, request, args, kwargs):
  356. if not instance.pk:
  357. # Assign the parent object based on URL kwargs
  358. content_type = get_object_or_404(ContentType, pk=request.GET.get('content_type'))
  359. instance.parent = get_object_or_404(content_type.model_class(), pk=request.GET.get('object_id'))
  360. return instance
  361. def get_return_url(self, request, obj=None):
  362. return obj.parent.get_absolute_url() if obj else super().get_return_url(request)
  363. class ImageAttachmentDeleteView(generic.ObjectDeleteView):
  364. queryset = ImageAttachment.objects.all()
  365. def get_return_url(self, request, obj=None):
  366. return obj.parent.get_absolute_url() if obj else super().get_return_url(request)
  367. #
  368. # Journal entries
  369. #
  370. class JournalEntryListView(generic.ObjectListView):
  371. queryset = JournalEntry.objects.all()
  372. filterset = filtersets.JournalEntryFilterSet
  373. filterset_form = forms.JournalEntryFilterForm
  374. table = tables.JournalEntryTable
  375. action_buttons = ('export',)
  376. class JournalEntryView(generic.ObjectView):
  377. queryset = JournalEntry.objects.all()
  378. class JournalEntryEditView(generic.ObjectEditView):
  379. queryset = JournalEntry.objects.all()
  380. model_form = forms.JournalEntryForm
  381. def alter_obj(self, obj, request, args, kwargs):
  382. if not obj.pk:
  383. obj.created_by = request.user
  384. return obj
  385. def get_return_url(self, request, instance):
  386. if not instance.assigned_object:
  387. return reverse('extras:journalentry_list')
  388. obj = instance.assigned_object
  389. viewname = f'{obj._meta.app_label}:{obj._meta.model_name}_journal'
  390. return reverse(viewname, kwargs={'pk': obj.pk})
  391. class JournalEntryDeleteView(generic.ObjectDeleteView):
  392. queryset = JournalEntry.objects.all()
  393. def get_return_url(self, request, instance):
  394. obj = instance.assigned_object
  395. viewname = f'{obj._meta.app_label}:{obj._meta.model_name}_journal'
  396. return reverse(viewname, kwargs={'pk': obj.pk})
  397. class JournalEntryBulkEditView(generic.BulkEditView):
  398. queryset = JournalEntry.objects.prefetch_related('created_by')
  399. filterset = filtersets.JournalEntryFilterSet
  400. table = tables.JournalEntryTable
  401. form = forms.JournalEntryBulkEditForm
  402. class JournalEntryBulkDeleteView(generic.BulkDeleteView):
  403. queryset = JournalEntry.objects.prefetch_related('created_by')
  404. filterset = filtersets.JournalEntryFilterSet
  405. table = tables.JournalEntryTable
  406. class ObjectJournalView(View):
  407. """
  408. Show all journal entries for an object.
  409. base_template: The name of the template to extend. If not provided, "<app>/<model>.html" will be used.
  410. """
  411. base_template = None
  412. def get(self, request, model, **kwargs):
  413. # Handle QuerySet restriction of parent object if needed
  414. if hasattr(model.objects, 'restrict'):
  415. obj = get_object_or_404(model.objects.restrict(request.user, 'view'), **kwargs)
  416. else:
  417. obj = get_object_or_404(model, **kwargs)
  418. # Gather all changes for this object (and its related objects)
  419. content_type = ContentType.objects.get_for_model(model)
  420. journalentries = JournalEntry.objects.restrict(request.user, 'view').prefetch_related('created_by').filter(
  421. assigned_object_type=content_type,
  422. assigned_object_id=obj.pk
  423. )
  424. journalentry_table = tables.ObjectJournalTable(journalentries)
  425. paginate_table(journalentry_table, request)
  426. if request.user.has_perm('extras.add_journalentry'):
  427. form = forms.JournalEntryForm(
  428. initial={
  429. 'assigned_object_type': ContentType.objects.get_for_model(obj),
  430. 'assigned_object_id': obj.pk
  431. }
  432. )
  433. else:
  434. form = None
  435. # Default to using "<app>/<model>.html" as the template, if it exists. Otherwise,
  436. # fall back to using base.html.
  437. if self.base_template is None:
  438. self.base_template = f"{model._meta.app_label}/{model._meta.model_name}.html"
  439. return render(request, 'extras/object_journal.html', {
  440. 'object': obj,
  441. 'form': form,
  442. 'table': journalentry_table,
  443. 'base_template': self.base_template,
  444. 'active_tab': 'journal',
  445. })
  446. #
  447. # Reports
  448. #
  449. class ReportListView(ContentTypePermissionRequiredMixin, View):
  450. """
  451. Retrieve all of the available reports from disk and the recorded JobResult (if any) for each.
  452. """
  453. def get_required_permission(self):
  454. return 'extras.view_report'
  455. def get(self, request):
  456. reports = get_reports()
  457. report_content_type = ContentType.objects.get(app_label='extras', model='report')
  458. results = {
  459. r.name: r
  460. for r in JobResult.objects.filter(
  461. obj_type=report_content_type,
  462. status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
  463. ).defer('data')
  464. }
  465. ret = []
  466. for module, report_list in reports:
  467. module_reports = []
  468. for report in report_list:
  469. report.result = results.get(report.full_name, None)
  470. module_reports.append(report)
  471. ret.append((module, module_reports))
  472. return render(request, 'extras/report_list.html', {
  473. 'reports': ret,
  474. })
  475. class ReportView(ContentTypePermissionRequiredMixin, View):
  476. """
  477. Display a single Report and its associated JobResult (if any).
  478. """
  479. def get_required_permission(self):
  480. return 'extras.view_report'
  481. def get(self, request, module, name):
  482. report = get_report(module, name)
  483. if report is None:
  484. raise Http404
  485. report_content_type = ContentType.objects.get(app_label='extras', model='report')
  486. report.result = JobResult.objects.filter(
  487. obj_type=report_content_type,
  488. name=report.full_name,
  489. status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
  490. ).first()
  491. return render(request, 'extras/report.html', {
  492. 'report': report,
  493. 'run_form': ConfirmationForm(),
  494. })
  495. def post(self, request, module, name):
  496. # Permissions check
  497. if not request.user.has_perm('extras.run_report'):
  498. return HttpResponseForbidden()
  499. report = get_report(module, name)
  500. if report is None:
  501. raise Http404
  502. # Allow execution only if RQ worker process is running
  503. if not Worker.count(get_connection('default')):
  504. messages.error(request, "Unable to run report: RQ worker process not running.")
  505. return render(request, 'extras/report.html', {
  506. 'report': report,
  507. })
  508. # Run the Report. A new JobResult is created.
  509. report_content_type = ContentType.objects.get(app_label='extras', model='report')
  510. job_result = JobResult.enqueue_job(
  511. run_report,
  512. report.full_name,
  513. report_content_type,
  514. request.user
  515. )
  516. return redirect('extras:report_result', job_result_pk=job_result.pk)
  517. class ReportResultView(ContentTypePermissionRequiredMixin, View):
  518. """
  519. Display a JobResult pertaining to the execution of a Report.
  520. """
  521. def get_required_permission(self):
  522. return 'extras.view_report'
  523. def get(self, request, job_result_pk):
  524. report_content_type = ContentType.objects.get(app_label='extras', model='report')
  525. result = get_object_or_404(JobResult.objects.all(), pk=job_result_pk, obj_type=report_content_type)
  526. # Retrieve the Report and attach the JobResult to it
  527. module, report_name = result.name.split('.')
  528. report = get_report(module, report_name)
  529. report.result = result
  530. # If this is an HTMX request, return only the result HTML
  531. if is_htmx(request):
  532. response = render(request, 'extras/htmx/report_result.html', {
  533. 'report': report,
  534. 'result': result,
  535. })
  536. if result.completed:
  537. response.status_code = 286
  538. return response
  539. return render(request, 'extras/report_result.html', {
  540. 'report': report,
  541. 'result': result,
  542. })
  543. #
  544. # Scripts
  545. #
  546. class GetScriptMixin:
  547. def _get_script(self, name, module=None):
  548. if module is None:
  549. module, name = name.split('.', 1)
  550. scripts = get_scripts()
  551. try:
  552. return scripts[module][name]()
  553. except KeyError:
  554. raise Http404
  555. class ScriptListView(ContentTypePermissionRequiredMixin, View):
  556. def get_required_permission(self):
  557. return 'extras.view_script'
  558. def get(self, request):
  559. scripts = get_scripts(use_names=True)
  560. script_content_type = ContentType.objects.get(app_label='extras', model='script')
  561. results = {
  562. r.name: r
  563. for r in JobResult.objects.filter(
  564. obj_type=script_content_type,
  565. status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
  566. ).defer('data')
  567. }
  568. for _scripts in scripts.values():
  569. for script in _scripts.values():
  570. script.result = results.get(script.full_name)
  571. return render(request, 'extras/script_list.html', {
  572. 'scripts': scripts,
  573. })
  574. class ScriptView(ContentTypePermissionRequiredMixin, GetScriptMixin, View):
  575. def get_required_permission(self):
  576. return 'extras.view_script'
  577. def get(self, request, module, name):
  578. script = self._get_script(name, module)
  579. form = script.as_form(initial=normalize_querydict(request.GET))
  580. # Look for a pending JobResult (use the latest one by creation timestamp)
  581. script_content_type = ContentType.objects.get(app_label='extras', model='script')
  582. script.result = JobResult.objects.filter(
  583. obj_type=script_content_type,
  584. name=script.full_name,
  585. ).exclude(
  586. status__in=JobResultStatusChoices.TERMINAL_STATE_CHOICES
  587. ).first()
  588. return render(request, 'extras/script.html', {
  589. 'module': module,
  590. 'script': script,
  591. 'form': form,
  592. })
  593. def post(self, request, module, name):
  594. # Permissions check
  595. if not request.user.has_perm('extras.run_script'):
  596. return HttpResponseForbidden()
  597. script = self._get_script(name, module)
  598. form = script.as_form(request.POST, request.FILES)
  599. # Allow execution only if RQ worker process is running
  600. if not Worker.count(get_connection('default')):
  601. messages.error(request, "Unable to run script: RQ worker process not running.")
  602. elif form.is_valid():
  603. commit = form.cleaned_data.pop('_commit')
  604. script_content_type = ContentType.objects.get(app_label='extras', model='script')
  605. job_result = JobResult.enqueue_job(
  606. run_script,
  607. script.full_name,
  608. script_content_type,
  609. request.user,
  610. data=form.cleaned_data,
  611. request=copy_safe_request(request),
  612. commit=commit
  613. )
  614. return redirect('extras:script_result', job_result_pk=job_result.pk)
  615. return render(request, 'extras/script.html', {
  616. 'module': module,
  617. 'script': script,
  618. 'form': form,
  619. })
  620. class ScriptResultView(ContentTypePermissionRequiredMixin, GetScriptMixin, View):
  621. def get_required_permission(self):
  622. return 'extras.view_script'
  623. def get(self, request, job_result_pk):
  624. result = get_object_or_404(JobResult.objects.all(), pk=job_result_pk)
  625. script_content_type = ContentType.objects.get(app_label='extras', model='script')
  626. if result.obj_type != script_content_type:
  627. raise Http404
  628. script = self._get_script(result.name)
  629. # If this is an HTMX request, return only the result HTML
  630. if is_htmx(request):
  631. response = render(request, 'extras/htmx/script_result.html', {
  632. 'script': script,
  633. 'result': result,
  634. })
  635. if result.completed:
  636. response.status_code = 286
  637. return response
  638. return render(request, 'extras/script_result.html', {
  639. 'script': script,
  640. 'result': result,
  641. 'class_name': script.__class__.__name__
  642. })