views.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451
  1. from django_tables2 import RequestConfig
  2. from django.contrib import messages
  3. from django.contrib.contenttypes.models import ContentType
  4. from django.core.exceptions import ImproperlyConfigured
  5. from django.core.urlresolvers import reverse
  6. from django.db import transaction, IntegrityError
  7. from django.db.models import ProtectedError
  8. from django.forms import ModelMultipleChoiceField, MultipleHiddenInput, TypedChoiceField
  9. from django.http import HttpResponse, HttpResponseRedirect
  10. from django.shortcuts import get_object_or_404, redirect, render
  11. from django.template import TemplateSyntaxError
  12. from django.utils.decorators import method_decorator
  13. from django.utils.http import is_safe_url
  14. from django.views.generic import View
  15. from extras.forms import CustomFieldForm
  16. from extras.models import CustomFieldValue, ExportTemplate, UserAction
  17. from .error_handlers import handle_protectederror
  18. from .forms import ConfirmationForm
  19. from .paginator import EnhancedPaginator
  20. class ObjectListView(View):
  21. queryset = None
  22. filter = None
  23. filter_form = None
  24. table = None
  25. edit_permissions = []
  26. template_name = None
  27. redirect_on_single_result = True
  28. def get(self, request, *args, **kwargs):
  29. model = self.queryset.model
  30. object_ct = ContentType.objects.get_for_model(model)
  31. if self.filter:
  32. self.queryset = self.filter(request.GET, self.queryset).qs
  33. # Check for export template rendering
  34. if request.GET.get('export'):
  35. et = get_object_or_404(ExportTemplate, content_type=object_ct, name=request.GET.get('export'))
  36. try:
  37. response = et.to_response(context_dict={'queryset': self.queryset.all()},
  38. filename='netbox_{}'.format(self.queryset.model._meta.verbose_name_plural))
  39. return response
  40. except TemplateSyntaxError:
  41. messages.error(request, "There was an error rendering the selected export template ({})."
  42. .format(et.name))
  43. # Fall back to built-in CSV export
  44. elif 'export' in request.GET and hasattr(model, 'to_csv'):
  45. output = '\n'.join([obj.to_csv() for obj in self.queryset.all()])
  46. response = HttpResponse(
  47. output,
  48. content_type='text/csv'
  49. )
  50. response['Content-Disposition'] = 'attachment; filename="netbox_{}.csv"'\
  51. .format(self.queryset.model._meta.verbose_name_plural)
  52. return response
  53. # Attempt to redirect automatically if the search query returns a single result
  54. if self.redirect_on_single_result and self.queryset.count() == 1 and request.GET:
  55. try:
  56. return HttpResponseRedirect(self.queryset[0].get_absolute_url())
  57. except AttributeError:
  58. pass
  59. # Provide a hook to tweak the queryset based on the request immediately prior to rendering the object list
  60. self.queryset = self.alter_queryset(request)
  61. # Construct the table based on the user's permissions
  62. table = self.table(self.queryset)
  63. table.model = model
  64. if 'pk' in table.base_columns and any([request.user.has_perm(perm) for perm in self.edit_permissions]):
  65. table.base_columns['pk'].visible = True
  66. RequestConfig(request, paginate={'klass': EnhancedPaginator}).configure(table)
  67. context = {
  68. 'table': table,
  69. 'filter_form': self.filter_form(request.GET, label_suffix='') if self.filter_form else None,
  70. 'export_templates': ExportTemplate.objects.filter(content_type=object_ct),
  71. }
  72. context.update(self.extra_context())
  73. return render(request, self.template_name, context)
  74. def alter_queryset(self, request):
  75. # .all() is necessary to avoid caching queries
  76. return self.queryset.all()
  77. def extra_context(self):
  78. return {}
  79. class ObjectEditView(View):
  80. model = None
  81. form_class = None
  82. fields_initial = []
  83. template_name = 'utilities/obj_edit.html'
  84. success_url = None
  85. cancel_url = None
  86. def get_object(self, kwargs):
  87. # Look up object by slug if one has been provided. Otherwise, use PK.
  88. if 'slug' in kwargs:
  89. return get_object_or_404(self.model, slug=kwargs['slug'])
  90. else:
  91. return get_object_or_404(self.model, pk=kwargs['pk'])
  92. def get(self, request, *args, **kwargs):
  93. if kwargs:
  94. obj = self.get_object(kwargs)
  95. form = self.form_class(instance=obj)
  96. else:
  97. obj = None
  98. form = self.form_class(initial={k: request.GET.get(k) for k in self.fields_initial})
  99. return render(request, self.template_name, {
  100. 'obj': obj,
  101. 'obj_type': self.model._meta.verbose_name,
  102. 'form': form,
  103. 'cancel_url': obj.get_absolute_url() if hasattr(obj, 'get_absolute_url') else reverse(self.cancel_url),
  104. })
  105. def post(self, request, *args, **kwargs):
  106. # Validate object if editing an existing object
  107. obj = self.get_object(kwargs) if kwargs else None
  108. form = self.form_class(request.POST, instance=obj)
  109. if form.is_valid():
  110. obj = form.save(commit=False)
  111. obj_created = not obj.pk
  112. obj.save()
  113. if isinstance(form, CustomFieldForm):
  114. form.save_custom_fields()
  115. msg = u'Created ' if obj_created else u'Modified '
  116. msg += self.model._meta.verbose_name
  117. if hasattr(obj, 'get_absolute_url'):
  118. msg = u'{} <a href="{}">{}</a>'.format(msg, obj.get_absolute_url(), obj)
  119. else:
  120. msg = u'{} {}'.format(msg, obj)
  121. messages.success(request, msg)
  122. if obj_created:
  123. UserAction.objects.log_create(request.user, obj, msg)
  124. else:
  125. UserAction.objects.log_edit(request.user, obj, msg)
  126. if '_addanother' in request.POST:
  127. return redirect(request.path)
  128. elif self.success_url:
  129. return redirect(self.success_url)
  130. else:
  131. return redirect(obj.get_absolute_url())
  132. return render(request, self.template_name, {
  133. 'obj': obj,
  134. 'obj_type': self.model._meta.verbose_name,
  135. 'form': form,
  136. 'cancel_url': obj.get_absolute_url() if hasattr(obj, 'get_absolute_url') else reverse(self.cancel_url),
  137. })
  138. class ObjectDeleteView(View):
  139. model = None
  140. template_name = 'utilities/obj_delete.html'
  141. redirect_url = None
  142. def get_object(self, kwargs):
  143. # Look up object by slug if one has been provided. Otherwise, use PK.
  144. if 'slug' in kwargs:
  145. return get_object_or_404(self.model, slug=kwargs['slug'])
  146. else:
  147. return get_object_or_404(self.model, pk=kwargs['pk'])
  148. def get(self, request, *args, **kwargs):
  149. obj = self.get_object(kwargs)
  150. form = ConfirmationForm()
  151. return render(request, self.template_name, {
  152. 'obj': obj,
  153. 'form': form,
  154. 'obj_type': self.model._meta.verbose_name,
  155. 'cancel_url': obj.get_absolute_url(),
  156. })
  157. def post(self, request, *args, **kwargs):
  158. obj = self.get_object(kwargs)
  159. form = ConfirmationForm(request.POST)
  160. if form.is_valid():
  161. try:
  162. obj.delete()
  163. msg = u'Deleted {} {}'.format(self.model._meta.verbose_name, obj)
  164. messages.success(request, msg)
  165. UserAction.objects.log_delete(request.user, obj, msg)
  166. return redirect(self.redirect_url)
  167. except ProtectedError, e:
  168. handle_protectederror(obj, request, e)
  169. return redirect(obj.get_absolute_url())
  170. return render(request, self.template_name, {
  171. 'obj': obj,
  172. 'form': form,
  173. 'obj_type': self.model._meta.verbose_name,
  174. 'cancel_url': obj.get_absolute_url(),
  175. })
  176. class BulkImportView(View):
  177. form = None
  178. table = None
  179. template_name = None
  180. obj_list_url = None
  181. def get(self, request, *args, **kwargs):
  182. return render(request, self.template_name, {
  183. 'form': self.form(),
  184. 'obj_list_url': self.obj_list_url,
  185. })
  186. def post(self, request, *args, **kwargs):
  187. form = self.form(request.POST)
  188. if form.is_valid():
  189. new_objs = []
  190. try:
  191. with transaction.atomic():
  192. for obj in form.cleaned_data['csv']:
  193. self.save_obj(obj)
  194. new_objs.append(obj)
  195. obj_table = self.table(new_objs)
  196. if new_objs:
  197. msg = u'Imported {} {}'.format(len(new_objs), new_objs[0]._meta.verbose_name_plural)
  198. messages.success(request, msg)
  199. UserAction.objects.log_import(request.user, ContentType.objects.get_for_model(new_objs[0]), msg)
  200. return render(request, "import_success.html", {
  201. 'table': obj_table,
  202. })
  203. except IntegrityError as e:
  204. form.add_error('csv', "Record {}: {}".format(len(new_objs) + 1, e.__cause__))
  205. return render(request, self.template_name, {
  206. 'form': form,
  207. 'obj_list_url': self.obj_list_url,
  208. })
  209. def save_obj(self, obj):
  210. obj.save()
  211. class BulkEditView(View):
  212. cls = None
  213. form = None
  214. template_name = None
  215. default_redirect_url = None
  216. def get(self, request, *args, **kwargs):
  217. return redirect(self.default_redirect_url)
  218. def post(self, request, *args, **kwargs):
  219. posted_redirect_url = request.POST.get('redirect_url')
  220. if posted_redirect_url and is_safe_url(url=posted_redirect_url, host=request.get_host()):
  221. redirect_url = posted_redirect_url
  222. else:
  223. redirect_url = reverse(self.default_redirect_url)
  224. if request.POST.get('_all'):
  225. pk_list = [int(pk) for pk in request.POST.get('pk_all').split(',') if pk]
  226. else:
  227. pk_list = [int(pk) for pk in request.POST.getlist('pk')]
  228. if '_apply' in request.POST:
  229. if hasattr(self.form, 'custom_fields'):
  230. form = self.form(self.cls, request.POST)
  231. else:
  232. form = self.form(request.POST)
  233. if form.is_valid():
  234. custom_fields = form.custom_fields if hasattr(form, 'custom_fields') else []
  235. standard_fields = [field for field in form.fields if field not in custom_fields and field != 'pk']
  236. # Update objects
  237. updated_count = self.update_objects(pk_list, form, standard_fields)
  238. # Update custom fields for objects
  239. if custom_fields:
  240. objs_updated = self.update_custom_fields(pk_list, form, custom_fields)
  241. if objs_updated and not updated_count:
  242. updated_count = objs_updated
  243. if updated_count:
  244. msg = u'Updated {} {}'.format(updated_count, self.cls._meta.verbose_name_plural)
  245. messages.success(self.request, msg)
  246. UserAction.objects.log_bulk_edit(request.user, ContentType.objects.get_for_model(self.cls), msg)
  247. return redirect(redirect_url)
  248. else:
  249. if hasattr(self.form, 'custom_fields'):
  250. form = self.form(self.cls, initial={'pk': pk_list})
  251. else:
  252. form = self.form(initial={'pk': pk_list})
  253. selected_objects = self.cls.objects.filter(pk__in=pk_list)
  254. if not selected_objects:
  255. messages.warning(request, "No {} were selected.".format(self.cls._meta.verbose_name_plural))
  256. return redirect(redirect_url)
  257. return render(request, self.template_name, {
  258. 'form': form,
  259. 'selected_objects': selected_objects,
  260. 'cancel_url': redirect_url,
  261. })
  262. def update_objects(self, pk_list, form, fields):
  263. fields_to_update = {}
  264. for name in fields:
  265. # Check for zero value (bulk editing)
  266. if isinstance(form.fields[name], TypedChoiceField) and form.cleaned_data[name] == 0:
  267. fields_to_update[name] = None
  268. elif form.cleaned_data[name]:
  269. fields_to_update[name] = form.cleaned_data[name]
  270. return self.cls.objects.filter(pk__in=pk_list).update(**fields_to_update)
  271. def update_custom_fields(self, pk_list, form, fields):
  272. obj_type = ContentType.objects.get_for_model(self.cls)
  273. objs_updated = False
  274. for name in fields:
  275. if form.cleaned_data[name] not in [None, u'']:
  276. field = form.fields[name].model
  277. # Check for zero value (bulk editing)
  278. if isinstance(form.fields[name], TypedChoiceField) and form.cleaned_data[name] == 0:
  279. serialized_value = field.serialize_value(None)
  280. else:
  281. serialized_value = field.serialize_value(form.cleaned_data[name])
  282. # Gather any pre-existing CustomFieldValues for the objects being edited.
  283. existing_cfvs = CustomFieldValue.objects.filter(field=field, obj_type=obj_type, obj_id__in=pk_list)
  284. # Determine which objects have an existing CFV to update and which need a new CFV created.
  285. update_list = [cfv['obj_id'] for cfv in existing_cfvs.values()]
  286. create_list = list(set(pk_list) - set(update_list))
  287. # Creating/updating CFVs
  288. if serialized_value:
  289. existing_cfvs.update(serialized_value=serialized_value)
  290. CustomFieldValue.objects.bulk_create([
  291. CustomFieldValue(field=field, obj_type=obj_type, obj_id=pk, serialized_value=serialized_value)
  292. for pk in create_list
  293. ])
  294. # Deleting CFVs
  295. else:
  296. existing_cfvs.delete()
  297. objs_updated = True
  298. return len(pk_list) if objs_updated else 0
  299. class BulkDeleteView(View):
  300. cls = None
  301. parent_cls = None
  302. form = None
  303. template_name = 'utilities/confirm_bulk_delete.html'
  304. default_redirect_url = None
  305. def post(self, request, *args, **kwargs):
  306. # Attempt to derive parent object if a parent class has been given
  307. if self.parent_cls:
  308. parent_obj = get_object_or_404(self.parent_cls, **kwargs)
  309. else:
  310. parent_obj = None
  311. # Determine URL to redirect users upon deletion of objects
  312. posted_redirect_url = request.POST.get('redirect_url')
  313. if posted_redirect_url and is_safe_url(url=posted_redirect_url, host=request.get_host()):
  314. redirect_url = posted_redirect_url
  315. elif parent_obj:
  316. redirect_url = parent_obj.get_absolute_url()
  317. elif self.default_redirect_url:
  318. redirect_url = reverse(self.default_redirect_url)
  319. else:
  320. raise ImproperlyConfigured('No redirect URL has been provided.')
  321. # Are we deleting *all* objects in the queryset or just a selected subset?
  322. if request.POST.get('_all'):
  323. pk_list = [x for x in request.POST.get('pk_all').split(',') if x]
  324. else:
  325. pk_list = request.POST.getlist('pk')
  326. form_cls = self.get_form()
  327. if '_confirm' in request.POST:
  328. form = form_cls(request.POST)
  329. if form.is_valid():
  330. # Delete objects
  331. queryset = self.cls.objects.filter(pk__in=pk_list)
  332. try:
  333. deleted_count = queryset.delete()[1][self.cls._meta.label]
  334. except ProtectedError, e:
  335. handle_protectederror(list(queryset), request, e)
  336. return redirect(redirect_url)
  337. msg = u'Deleted {} {}'.format(deleted_count, self.cls._meta.verbose_name_plural)
  338. messages.success(request, msg)
  339. UserAction.objects.log_bulk_delete(request.user, ContentType.objects.get_for_model(self.cls), msg)
  340. return redirect(redirect_url)
  341. else:
  342. form = form_cls(initial={'pk': pk_list})
  343. selected_objects = self.cls.objects.filter(pk__in=pk_list)
  344. if not selected_objects:
  345. messages.warning(request, "No {} were selected for deletion.".format(self.cls._meta.verbose_name_plural))
  346. return redirect(redirect_url)
  347. return render(request, self.template_name, {
  348. 'form': form,
  349. 'parent_obj': parent_obj,
  350. 'obj_type_plural': self.cls._meta.verbose_name_plural,
  351. 'selected_objects': selected_objects,
  352. 'cancel_url': redirect_url,
  353. })
  354. def get_form(self):
  355. """Provide a standard bulk delete form if none has been specified for the view"""
  356. class BulkDeleteForm(ConfirmationForm):
  357. pk = ModelMultipleChoiceField(queryset=self.cls.objects.all(), widget=MultipleHiddenInput)
  358. if self.form:
  359. return self.form
  360. return BulkDeleteForm