views.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325
  1. from django.conf import settings
  2. from django.contrib import messages
  3. from django.contrib.admin.views.decorators import staff_member_required
  4. from django.contrib.contenttypes.models import ContentType
  5. from django.core.urlresolvers import reverse, NoReverseMatch
  6. from django.db import transaction, IntegrityError
  7. from django.db.models import ProtectedError
  8. from django.http import HttpResponseRedirect
  9. from django.shortcuts import get_object_or_404, redirect, render
  10. from django.template import TemplateSyntaxError
  11. from django.utils.decorators import method_decorator
  12. from django.utils.http import is_safe_url
  13. from django.views.generic import View
  14. from django_tables2 import RequestConfig
  15. from .error_handlers import handle_protectederror
  16. from .forms import ConfirmationForm
  17. from .paginator import EnhancedPaginator
  18. from extras.models import ExportTemplate
  19. class ObjectListView(View):
  20. queryset = None
  21. filter = None
  22. filter_form = None
  23. table = None
  24. edit_permissions = []
  25. template_name = None
  26. redirect_on_single_result = True
  27. def get(self, request, *args, **kwargs):
  28. model = self.queryset.model
  29. object_ct = ContentType.objects.get_for_model(model)
  30. if self.filter:
  31. self.queryset = self.filter(request.GET, self.queryset).qs
  32. # Check for export template rendering
  33. if request.GET.get('export'):
  34. et = get_object_or_404(ExportTemplate, content_type=object_ct, name=request.GET.get('export'))
  35. try:
  36. response = et.to_response(context_dict={'queryset': self.queryset},
  37. filename='netbox_{}'.format(self.queryset.model._meta.verbose_name_plural))
  38. return response
  39. except TemplateSyntaxError:
  40. messages.error(request, "There was an error rendering the selected export template ({}).".format(et.name))
  41. # Attempt to redirect automatically if the query returns a single result
  42. if self.redirect_on_single_result and self.queryset.count() == 1:
  43. try:
  44. return HttpResponseRedirect(self.queryset[0].get_absolute_url())
  45. except AttributeError:
  46. pass
  47. # Provide a hook to tweak the queryset based on the request immediately prior to rendering the object list
  48. self.queryset = self.alter_queryset(request)
  49. # Construct the table based on the user's permissions
  50. table = self.table(self.queryset)
  51. table.model = model
  52. if 'pk' in table.base_columns and any([request.user.has_perm(perm) for perm in self.edit_permissions]):
  53. table.base_columns['pk'].visible = True
  54. RequestConfig(request, paginate={'per_page': settings.PAGINATE_COUNT, 'klass': EnhancedPaginator})\
  55. .configure(table)
  56. return render(request, self.template_name, {
  57. 'table': table,
  58. 'filter_form': self.filter_form(request.GET, label_suffix='') if self.filter_form else None,
  59. 'export_templates': ExportTemplate.objects.filter(content_type=object_ct),
  60. })
  61. def alter_queryset(self, request):
  62. return self.queryset
  63. class ObjectEditView(View):
  64. model = None
  65. form_class = None
  66. fields_initial = []
  67. template_name = 'utilities/obj_edit.html'
  68. success_url = None
  69. cancel_url = None
  70. def get_object(self, kwargs):
  71. # Look up object by slug if one has been provided. Otherwise, use PK.
  72. if 'slug' in kwargs:
  73. return get_object_or_404(self.model, slug=kwargs['slug'])
  74. else:
  75. return get_object_or_404(self.model, pk=kwargs['pk'])
  76. def get(self, request, *args, **kwargs):
  77. if kwargs:
  78. obj = self.get_object(kwargs)
  79. form = self.form_class(instance=obj)
  80. else:
  81. obj = None
  82. form = self.form_class(initial={k: request.GET.get(k) for k in self.fields_initial})
  83. return render(request, self.template_name, {
  84. 'obj': obj,
  85. 'obj_type': self.model._meta.verbose_name,
  86. 'form': form,
  87. 'cancel_url': reverse(self.cancel_url) if self.cancel_url else obj.get_absolute_url(),
  88. })
  89. def post(self, request, *args, **kwargs):
  90. # Validate object if editing an existing object
  91. obj = self.get_object(kwargs) if kwargs else None
  92. form = self.form_class(request.POST, instance=obj)
  93. if form.is_valid():
  94. obj = form.save(commit=False)
  95. obj_created = not obj.pk
  96. obj.save()
  97. msg = 'Created ' if obj_created else 'Modified '
  98. msg += self.model._meta.verbose_name
  99. if hasattr(obj, 'get_absolute_url'):
  100. msg += ' <a href="{}">{}</a>'.format(obj.get_absolute_url(), obj)
  101. else:
  102. msg += ' {}'.format(obj)
  103. messages.success(request, msg)
  104. if '_addanother' in request.POST:
  105. return redirect(request.path)
  106. elif self.success_url:
  107. return redirect(self.success_url)
  108. else:
  109. return redirect(obj.get_absolute_url())
  110. return render(request, self.template_name, {
  111. 'obj': obj,
  112. 'obj_type': self.model._meta.verbose_name,
  113. 'form': form,
  114. 'cancel_url': reverse(self.cancel_url) if self.cancel_url else obj.get_absolute_url(),
  115. })
  116. class ObjectDeleteView(View):
  117. model = None
  118. template_name = 'utilities/obj_delete.html'
  119. redirect_url = None
  120. def get_object(self, kwargs):
  121. # Look up object by slug if one has been provided. Otherwise, use PK.
  122. if 'slug' in kwargs:
  123. return get_object_or_404(self.model, slug=kwargs['slug'])
  124. else:
  125. return get_object_or_404(self.model, pk=kwargs['pk'])
  126. def get(self, request, *args, **kwargs):
  127. obj = self.get_object(kwargs)
  128. form = ConfirmationForm()
  129. return render(request, self.template_name, {
  130. 'obj': obj,
  131. 'form': form,
  132. 'obj_type': self.model._meta.verbose_name,
  133. 'cancel_url': obj.get_absolute_url(),
  134. })
  135. def post(self, request, *args, **kwargs):
  136. obj = self.get_object(kwargs)
  137. form = ConfirmationForm(request.POST)
  138. if form.is_valid():
  139. try:
  140. obj.delete()
  141. messages.success(request, 'Deleted {} {}'.format(self.model._meta.verbose_name, obj))
  142. return redirect(self.redirect_url)
  143. except ProtectedError, e:
  144. handle_protectederror(obj, request, e)
  145. return redirect(obj.get_absolute_url())
  146. return render(request, self.template_name, {
  147. 'obj': obj,
  148. 'form': form,
  149. 'obj_type': self.model._meta.verbose_name,
  150. 'cancel_url': obj.get_absolute_url(),
  151. })
  152. class BulkImportView(View):
  153. form = None
  154. table = None
  155. template_name = None
  156. obj_list_url = None
  157. def get(self, request, *args, **kwargs):
  158. return render(request, self.template_name, {
  159. 'form': self.form(),
  160. 'obj_list_url': self.obj_list_url,
  161. })
  162. def post(self, request, *args, **kwargs):
  163. form = self.form(request.POST)
  164. if form.is_valid():
  165. new_objs = []
  166. try:
  167. with transaction.atomic():
  168. for obj in form.cleaned_data['csv']:
  169. self.save_obj(obj)
  170. new_objs.append(obj)
  171. obj_table = self.table(new_objs)
  172. messages.success(request, "Imported {} objects".format(len(new_objs)))
  173. return render(request, "import_success.html", {
  174. 'table': obj_table,
  175. })
  176. except IntegrityError as e:
  177. form.add_error('csv', "Record {}: {}".format(len(new_objs) + 1, e.__cause__))
  178. return render(request, self.template_name, {
  179. 'form': form,
  180. 'obj_list_url': self.obj_list_url,
  181. })
  182. def save_obj(self, obj):
  183. obj.save()
  184. class BulkEditView(View):
  185. cls = None
  186. form = None
  187. template_name = None
  188. default_redirect_url = None
  189. def get(self, request, *args, **kwargs):
  190. return redirect(self.default_redirect_url)
  191. def post(self, request, *args, **kwargs):
  192. posted_redirect_url = request.POST.get('redirect_url')
  193. if posted_redirect_url and is_safe_url(url=posted_redirect_url, host=request.get_host()):
  194. redirect_url = posted_redirect_url
  195. else:
  196. redirect_url = reverse(self.default_redirect_url)
  197. if '_apply' in request.POST:
  198. form = self.form(request.POST)
  199. if form.is_valid():
  200. pk_list = [obj.pk for obj in form.cleaned_data['pk']]
  201. self.update_objects(pk_list, form)
  202. if not form.errors:
  203. return redirect(redirect_url)
  204. else:
  205. form = self.form(initial={'pk': request.POST.getlist('pk')})
  206. selected_objects = self.cls.objects.filter(pk__in=request.POST.getlist('pk'))
  207. if not selected_objects:
  208. messages.warning(request, "No {} were selected.".format(self.cls._meta.verbose_name_plural))
  209. return redirect(redirect_url)
  210. return render(request, self.template_name, {
  211. 'form': form,
  212. 'selected_objects': selected_objects,
  213. 'cancel_url': redirect_url,
  214. })
  215. def update_objects(self, obj_list, form):
  216. """
  217. This method provides the update logic (must be overridden by subclasses).
  218. """
  219. raise NotImplementedError()
  220. class BulkDeleteView(View):
  221. cls = None
  222. form = None
  223. template_name = 'utilities/confirm_bulk_delete.html'
  224. default_redirect_url = None
  225. @method_decorator(staff_member_required)
  226. def dispatch(self, *args, **kwargs):
  227. return super(BulkDeleteView, self).dispatch(*args, **kwargs)
  228. def get(self, request, *args, **kwargs):
  229. return redirect(self.default_redirect_url)
  230. def post(self, request, *args, **kwargs):
  231. posted_redirect_url = request.POST.get('redirect_url')
  232. if posted_redirect_url and is_safe_url(url=posted_redirect_url, host=request.get_host()):
  233. redirect_url = posted_redirect_url
  234. else:
  235. redirect_url = reverse(self.default_redirect_url)
  236. if '_confirm' in request.POST:
  237. form = self.form(request.POST)
  238. if form.is_valid():
  239. # Delete objects
  240. objects_to_delete = self.cls.objects.filter(pk__in=[v.id for v in form.cleaned_data['pk']])
  241. try:
  242. deleted_count = objects_to_delete.count()
  243. objects_to_delete.delete()
  244. except ProtectedError, e:
  245. handle_protectederror(list(objects_to_delete), request, e)
  246. return redirect(redirect_url)
  247. messages.success(request, "Deleted {} {}".format(deleted_count, self.cls._meta.verbose_name_plural))
  248. return redirect(redirect_url)
  249. else:
  250. form = self.form(initial={'pk': request.POST.getlist('pk')})
  251. selected_objects = self.cls.objects.filter(pk__in=form.initial.get('pk'))
  252. if not selected_objects:
  253. messages.warning(request, "No {} were selected for deletion.".format(self.cls._meta.verbose_name_plural))
  254. return redirect(redirect_url)
  255. return render(request, self.template_name, {
  256. 'form': form,
  257. 'obj_type_plural': self.cls._meta.verbose_name_plural,
  258. 'selected_objects': selected_objects,
  259. 'cancel_url': redirect_url,
  260. })