views.py 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. from __future__ import unicode_literals
  2. from django import template
  3. from django.contrib import messages
  4. from django.contrib.auth.mixins import PermissionRequiredMixin
  5. from django.contrib.contenttypes.models import ContentType
  6. from django.db.models import Count, Q
  7. from django.http import Http404
  8. from django.shortcuts import get_object_or_404, redirect, render, reverse
  9. from django.utils.safestring import mark_safe
  10. from django.views.generic import View
  11. from taggit.models import Tag
  12. from utilities.forms import ConfirmationForm
  13. from utilities.views import BulkDeleteView, ObjectDeleteView, ObjectEditView, ObjectListView
  14. from . import filters
  15. from .forms import ObjectChangeFilterForm, ImageAttachmentForm, TagForm
  16. from .models import ImageAttachment, ObjectChange, ReportResult
  17. from .reports import get_report, get_reports
  18. from .tables import ObjectChangeTable, TagTable
  19. #
  20. # Tags
  21. #
  22. class TagListView(ObjectListView):
  23. queryset = Tag.objects.annotate(items=Count('taggit_taggeditem_items')).order_by('name')
  24. table = TagTable
  25. template_name = 'extras/tag_list.html'
  26. class TagEditView(PermissionRequiredMixin, ObjectEditView):
  27. permission_required = 'taggit.change_tag'
  28. model = Tag
  29. model_form = TagForm
  30. def get_return_url(self, request, obj):
  31. return reverse('extras:tag', kwargs={'slug': obj.slug})
  32. class TagDeleteView(PermissionRequiredMixin, ObjectDeleteView):
  33. permission_required = 'taggit.delete_tag'
  34. model = Tag
  35. default_return_url = 'extras:tag_list'
  36. class TagBulkDeleteView(PermissionRequiredMixin, BulkDeleteView):
  37. permission_required = 'circuits.delete_circuittype'
  38. cls = Tag
  39. queryset = Tag.objects.annotate(items=Count('taggit_taggeditem_items')).order_by('name')
  40. table = TagTable
  41. default_return_url = 'extras:tag_list'
  42. #
  43. # Change logging
  44. #
  45. class ObjectChangeListView(ObjectListView):
  46. queryset = ObjectChange.objects.select_related('user', 'changed_object_type')
  47. filter = filters.ObjectChangeFilter
  48. filter_form = ObjectChangeFilterForm
  49. table = ObjectChangeTable
  50. template_name = 'extras/objectchange_list.html'
  51. class ObjectChangeView(View):
  52. def get(self, request, pk):
  53. objectchange = get_object_or_404(ObjectChange, pk=pk)
  54. related_changes = ObjectChange.objects.filter(request_id=objectchange.request_id).exclude(pk=objectchange.pk)
  55. related_changes_table = ObjectChangeTable(
  56. data=related_changes[:50],
  57. orderable=False
  58. )
  59. return render(request, 'extras/objectchange.html', {
  60. 'objectchange': objectchange,
  61. 'related_changes_table': related_changes_table,
  62. 'related_changes_count': related_changes.count()
  63. })
  64. class ObjectChangeLogView(View):
  65. """
  66. Present a history of changes made to a particular object.
  67. """
  68. def get(self, request, model, **kwargs):
  69. # Get object my model and kwargs (e.g. slug='foo')
  70. obj = get_object_or_404(model, **kwargs)
  71. # Gather all changes for this object (and its related objects)
  72. content_type = ContentType.objects.get_for_model(model)
  73. objectchanges = ObjectChange.objects.select_related(
  74. 'user', 'changed_object_type'
  75. ).filter(
  76. Q(changed_object_type=content_type, changed_object_id=obj.pk) |
  77. Q(related_object_type=content_type, related_object_id=obj.pk)
  78. )
  79. objectchanges_table = ObjectChangeTable(
  80. data=objectchanges,
  81. orderable=False
  82. )
  83. # Check whether a header template exists for this model
  84. base_template = '{}/{}.html'.format(model._meta.app_label, model._meta.model_name)
  85. try:
  86. template.loader.get_template(base_template)
  87. object_var = model._meta.model_name
  88. except template.TemplateDoesNotExist:
  89. base_template = '_base.html'
  90. object_var = 'obj'
  91. return render(request, 'extras/object_changelog.html', {
  92. object_var: obj,
  93. 'objectchanges_table': objectchanges_table,
  94. 'base_template': base_template,
  95. 'active_tab': 'changelog',
  96. })
  97. #
  98. # Image attachments
  99. #
  100. class ImageAttachmentEditView(PermissionRequiredMixin, ObjectEditView):
  101. permission_required = 'extras.change_imageattachment'
  102. model = ImageAttachment
  103. model_form = ImageAttachmentForm
  104. def alter_obj(self, imageattachment, request, args, kwargs):
  105. if not imageattachment.pk:
  106. # Assign the parent object based on URL kwargs
  107. model = kwargs.get('model')
  108. imageattachment.parent = get_object_or_404(model, pk=kwargs['object_id'])
  109. return imageattachment
  110. def get_return_url(self, request, imageattachment):
  111. return imageattachment.parent.get_absolute_url()
  112. class ImageAttachmentDeleteView(PermissionRequiredMixin, ObjectDeleteView):
  113. permission_required = 'extras.delete_imageattachment'
  114. model = ImageAttachment
  115. def get_return_url(self, request, imageattachment):
  116. return imageattachment.parent.get_absolute_url()
  117. #
  118. # Reports
  119. #
  120. class ReportListView(View):
  121. """
  122. Retrieve all of the available reports from disk and the recorded ReportResult (if any) for each.
  123. """
  124. def get(self, request):
  125. reports = get_reports()
  126. results = {r.report: r for r in ReportResult.objects.all()}
  127. ret = []
  128. for module, report_list in reports:
  129. module_reports = []
  130. for report in report_list:
  131. report.result = results.get(report.full_name, None)
  132. module_reports.append(report)
  133. ret.append((module, module_reports))
  134. return render(request, 'extras/report_list.html', {
  135. 'reports': ret,
  136. })
  137. class ReportView(View):
  138. """
  139. Display a single Report and its associated ReportResult (if any).
  140. """
  141. def get(self, request, name):
  142. # Retrieve the Report by "<module>.<report>"
  143. module_name, report_name = name.split('.')
  144. report = get_report(module_name, report_name)
  145. if report is None:
  146. raise Http404
  147. # Attach the ReportResult (if any)
  148. report.result = ReportResult.objects.filter(report=report.full_name).first()
  149. return render(request, 'extras/report.html', {
  150. 'report': report,
  151. 'run_form': ConfirmationForm(),
  152. })
  153. class ReportRunView(PermissionRequiredMixin, View):
  154. """
  155. Run a Report and record a new ReportResult.
  156. """
  157. permission_required = 'extras.add_reportresult'
  158. def post(self, request, name):
  159. # Retrieve the Report by "<module>.<report>"
  160. module_name, report_name = name.split('.')
  161. report = get_report(module_name, report_name)
  162. if report is None:
  163. raise Http404
  164. form = ConfirmationForm(request.POST)
  165. if form.is_valid():
  166. # Run the Report. A new ReportResult is created.
  167. report.run()
  168. result = 'failed' if report.failed else 'passed'
  169. msg = "Ran report {} ({})".format(report.full_name, result)
  170. messages.success(request, mark_safe(msg))
  171. return redirect('extras:report', name=report.full_name)