views.py 13 KB

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