views.py 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. import logging
  2. from django.conf import settings
  3. from django.contrib import messages
  4. from django.contrib.auth import login as auth_login, logout as auth_logout, update_session_auth_hash
  5. from django.contrib.auth.mixins import LoginRequiredMixin
  6. from django.contrib.auth.models import update_last_login
  7. from django.contrib.auth.signals import user_logged_in
  8. from django.http import HttpResponseRedirect
  9. from django.shortcuts import get_object_or_404, redirect, render
  10. from django.urls import reverse
  11. from django.utils.decorators import method_decorator
  12. from django.utils.http import is_safe_url
  13. from django.views.decorators.debug import sensitive_post_parameters
  14. from django.views.generic import View
  15. from utilities.forms import ConfirmationForm
  16. from .forms import LoginForm, PasswordChangeForm, TokenForm
  17. from .models import Token
  18. #
  19. # Login/logout
  20. #
  21. class LoginView(View):
  22. """
  23. Perform user authentication via the web UI.
  24. """
  25. template_name = 'login.html'
  26. @method_decorator(sensitive_post_parameters('password'))
  27. def dispatch(self, *args, **kwargs):
  28. return super().dispatch(*args, **kwargs)
  29. def get(self, request):
  30. form = LoginForm(request)
  31. if request.user.is_authenticated:
  32. logger = logging.getLogger('netbox.auth.login')
  33. return self.redirect_to_next(request, logger)
  34. return render(request, self.template_name, {
  35. 'form': form,
  36. })
  37. def post(self, request):
  38. logger = logging.getLogger('netbox.auth.login')
  39. form = LoginForm(request, data=request.POST)
  40. if form.is_valid():
  41. logger.debug("Login form validation was successful")
  42. # If maintenance mode is enabled, assume the database is read-only, and disable updating the user's
  43. # last_login time upon authentication.
  44. if settings.MAINTENANCE_MODE:
  45. logger.warning("Maintenance mode enabled: disabling update of most recent login time")
  46. user_logged_in.disconnect(update_last_login, dispatch_uid='update_last_login')
  47. # Authenticate user
  48. auth_login(request, form.get_user())
  49. logger.info(f"User {request.user} successfully authenticated")
  50. messages.info(request, "Logged in as {}.".format(request.user))
  51. return self.redirect_to_next(request, logger)
  52. else:
  53. logger.debug("Login form validation failed")
  54. return render(request, self.template_name, {
  55. 'form': form,
  56. })
  57. def redirect_to_next(self, request, logger):
  58. if request.method == "POST":
  59. redirect_to = request.POST.get('next', reverse('home'))
  60. else:
  61. redirect_to = request.GET.get('next', reverse('home'))
  62. if redirect_to and not is_safe_url(url=redirect_to, allowed_hosts=request.get_host()):
  63. logger.warning(f"Ignoring unsafe 'next' URL passed to login form: {redirect_to}")
  64. redirect_to = reverse('home')
  65. logger.debug(f"Redirecting user to {redirect_to}")
  66. return HttpResponseRedirect(redirect_to)
  67. class LogoutView(View):
  68. """
  69. Deauthenticate a web user.
  70. """
  71. def get(self, request):
  72. logger = logging.getLogger('netbox.auth.logout')
  73. # Log out the user
  74. username = request.user
  75. auth_logout(request)
  76. logger.info(f"User {username} has logged out")
  77. messages.info(request, "You have logged out.")
  78. # Delete session key cookie (if set) upon logout
  79. response = HttpResponseRedirect(reverse('home'))
  80. response.delete_cookie('session_key')
  81. return response
  82. #
  83. # User profiles
  84. #
  85. class ProfileView(LoginRequiredMixin, View):
  86. template_name = 'users/profile.html'
  87. def get(self, request):
  88. return render(request, self.template_name, {
  89. 'active_tab': 'profile',
  90. })
  91. class UserConfigView(LoginRequiredMixin, View):
  92. template_name = 'users/preferences.html'
  93. def get(self, request):
  94. return render(request, self.template_name, {
  95. 'preferences': request.user.config.all(),
  96. 'active_tab': 'preferences',
  97. })
  98. def post(self, request):
  99. userconfig = request.user.config
  100. data = userconfig.all()
  101. # Delete selected preferences
  102. if "_delete" in request.POST:
  103. for key in request.POST.getlist('pk'):
  104. if key in data:
  105. userconfig.clear(key)
  106. # Update specific values
  107. elif "_update" in request.POST:
  108. for key in request.POST:
  109. if not key.startswith('_') and not key.startswith('csrf'):
  110. for value in request.POST.getlist(key):
  111. userconfig.set(key, value)
  112. userconfig.save()
  113. messages.success(request, "Your preferences have been updated.")
  114. return redirect('user:preferences')
  115. class ChangePasswordView(LoginRequiredMixin, View):
  116. template_name = 'users/password.html'
  117. def get(self, request):
  118. # LDAP users cannot change their password here
  119. if getattr(request.user, 'ldap_username', None):
  120. messages.warning(request, "LDAP-authenticated user credentials cannot be changed within NetBox.")
  121. return redirect('user:profile')
  122. form = PasswordChangeForm(user=request.user)
  123. return render(request, self.template_name, {
  124. 'form': form,
  125. 'active_tab': 'password',
  126. })
  127. def post(self, request):
  128. form = PasswordChangeForm(user=request.user, data=request.POST)
  129. if form.is_valid():
  130. form.save()
  131. update_session_auth_hash(request, form.user)
  132. messages.success(request, "Your password has been changed successfully.")
  133. return redirect('user:profile')
  134. return render(request, self.template_name, {
  135. 'form': form,
  136. 'active_tab': 'change_password',
  137. })
  138. #
  139. # API tokens
  140. #
  141. class TokenListView(LoginRequiredMixin, View):
  142. def get(self, request):
  143. tokens = Token.objects.filter(user=request.user)
  144. return render(request, 'users/api_tokens.html', {
  145. 'tokens': tokens,
  146. 'active_tab': 'api-tokens',
  147. })
  148. class TokenEditView(LoginRequiredMixin, View):
  149. def get(self, request, pk=None):
  150. if pk:
  151. token = get_object_or_404(Token.objects.filter(user=request.user), pk=pk)
  152. else:
  153. token = Token(user=request.user)
  154. form = TokenForm(instance=token)
  155. return render(request, 'generic/object_edit.html', {
  156. 'obj': token,
  157. 'obj_type': token._meta.verbose_name,
  158. 'form': form,
  159. 'return_url': reverse('user:token_list'),
  160. })
  161. def post(self, request, pk=None):
  162. if pk:
  163. token = get_object_or_404(Token.objects.filter(user=request.user), pk=pk)
  164. form = TokenForm(request.POST, instance=token)
  165. else:
  166. token = Token(user=request.user)
  167. form = TokenForm(request.POST)
  168. if form.is_valid():
  169. token = form.save(commit=False)
  170. token.user = request.user
  171. token.save()
  172. msg = f"Modified token {token}" if pk else f"Created token {token}"
  173. messages.success(request, msg)
  174. if '_addanother' in request.POST:
  175. return redirect(request.path)
  176. else:
  177. return redirect('user:token_list')
  178. return render(request, 'generic/object_edit.html', {
  179. 'obj': token,
  180. 'obj_type': token._meta.verbose_name,
  181. 'form': form,
  182. 'return_url': reverse('user:token_list'),
  183. })
  184. class TokenDeleteView(LoginRequiredMixin, View):
  185. def get(self, request, pk):
  186. token = get_object_or_404(Token.objects.filter(user=request.user), pk=pk)
  187. initial_data = {
  188. 'return_url': reverse('user:token_list'),
  189. }
  190. form = ConfirmationForm(initial=initial_data)
  191. return render(request, 'generic/object_delete.html', {
  192. 'obj': token,
  193. 'obj_type': token._meta.verbose_name,
  194. 'form': form,
  195. 'return_url': reverse('user:token_list'),
  196. })
  197. def post(self, request, pk):
  198. token = get_object_or_404(Token.objects.filter(user=request.user), pk=pk)
  199. form = ConfirmationForm(request.POST)
  200. if form.is_valid():
  201. token.delete()
  202. messages.success(request, "Token deleted")
  203. return redirect('user:token_list')
  204. return render(request, 'generic/object_delete.html', {
  205. 'obj': token,
  206. 'obj_type': token._meta.verbose_name,
  207. 'form': form,
  208. 'return_url': reverse('user:token_list'),
  209. })