pipeline.py 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. import json
  2. from django.contrib.auth import get_user_model
  3. from django.db import IntegrityError
  4. from django.http import JsonResponse
  5. from django.shortcuts import render
  6. from django.urls import reverse
  7. from django.utils.translation import ugettext as _
  8. from social_core.pipeline.partial import partial
  9. from misago.conf import settings
  10. from misago.core.exceptions import SocialAuthFailed, SocialAuthBanned
  11. from misago.legal.models import Agreement
  12. from misago.users.bans import get_request_ip_ban, get_user_ban
  13. from misago.users.forms.register import SocialAuthRegisterForm
  14. from misago.users.models import Ban
  15. from misago.users.registration import (
  16. get_registration_result_json, save_user_agreements, send_welcome_email
  17. )
  18. from misago.users.validators import (
  19. ValidationError, validate_new_registration, validate_email, validate_username)
  20. from .utils import get_social_auth_backend_name, perpare_username
  21. UserModel = get_user_model()
  22. def validate_ip_not_banned(strategy, details, backend, user=None, *args, **kwargs):
  23. """Pipeline step that interrupts pipeline if found user is non-staff and IP banned"""
  24. if not user or user.is_staff:
  25. return None
  26. ban = get_request_ip_ban(strategy.request)
  27. if ban:
  28. hydrated_ban = Ban(
  29. check_type=Ban.IP,
  30. user_message=ban['message'],
  31. expires_on=ban['expires_on'],
  32. )
  33. raise SocialAuthBanned(backend, hydrated_ban)
  34. def validate_user_not_banned(strategy, details, backend, user=None, *args, **kwargs):
  35. """Pipeline step that interrupts pipeline if found user is non-staff and banned"""
  36. if not user or user.is_staff:
  37. return None
  38. user_ban = get_user_ban(user)
  39. if user_ban:
  40. raise SocialAuthBanned(backend, user_ban)
  41. def associate_by_email(strategy, details, backend, user=None, *args, **kwargs):
  42. """If user with e-mail from provider exists in database and is active,
  43. this step authenticates them.
  44. """
  45. if user:
  46. return None
  47. email = details.get('email')
  48. if not email:
  49. return None
  50. try:
  51. user = UserModel.objects.get_by_email(email)
  52. except UserModel.DoesNotExist:
  53. return None
  54. backend_name = get_social_auth_backend_name(backend.name)
  55. if not user.is_active:
  56. raise SocialAuthFailed(
  57. backend,
  58. _(
  59. "The e-mail address associated with your %(backend)s account is "
  60. "not available for use on this site."
  61. ) % {'backend': backend_name}
  62. )
  63. if user.requires_activation_by_admin:
  64. raise SocialAuthFailed(
  65. backend,
  66. _(
  67. "Your account has to be activated by site administrator before you will be able to "
  68. "sign in with %(backend)s."
  69. ) % {'backend': backend_name}
  70. )
  71. return {'user': user, 'is_new': False}
  72. def get_username(strategy, details, backend, user=None, *args, **kwargs):
  73. """Resolve valid username for use in new account"""
  74. if user:
  75. return None
  76. username = perpare_username(details.get('username', ''))
  77. full_name = perpare_username(details.get('full_name', ''))
  78. first_name = perpare_username(details.get('first_name', ''))
  79. last_name = perpare_username(details.get('last_name', ''))
  80. names_to_try = [
  81. username,
  82. first_name,
  83. ]
  84. if username:
  85. names_to_try.append(username)
  86. if first_name:
  87. names_to_try.append(first_name)
  88. if last_name:
  89. # if first name is taken, try first name + first char of last name
  90. names_to_try.append(first_name + last_name[0])
  91. if full_name:
  92. names_to_try.append(full_name)
  93. username_length_max = settings.username_length_max
  94. for name in names_to_try:
  95. if len(name) > username_length_max:
  96. names_to_try.append(name[:username_length_max])
  97. for name in filter(bool, names_to_try):
  98. try:
  99. validate_username(name)
  100. return {'clean_username': name}
  101. except ValidationError:
  102. pass
  103. def create_user(strategy, details, backend, user=None, *args, **kwargs):
  104. """Aggressively attempt to register and sign in new user"""
  105. if user:
  106. return None
  107. request = strategy.request
  108. email = details.get('email')
  109. username = kwargs.get('clean_username')
  110. if not email or not username:
  111. return None
  112. try:
  113. validate_email(email)
  114. validate_new_registration(request, {
  115. 'email': email,
  116. 'username': username,
  117. })
  118. except ValidationError:
  119. return None
  120. activation_kwargs = {}
  121. if settings.account_activation == 'admin':
  122. activation_kwargs = {'requires_activation': UserModel.ACTIVATION_ADMIN}
  123. new_user = UserModel.objects.create_user(
  124. username,
  125. email,
  126. create_audit_trail=True,
  127. joined_from_ip=request.user_ip,
  128. set_default_avatar=True,
  129. **activation_kwargs
  130. )
  131. send_welcome_email(request, new_user)
  132. return {'user': new_user, 'is_new': True}
  133. @partial
  134. def create_user_with_form(strategy, details, backend, user=None, *args, **kwargs):
  135. """Alternatively to create_user lets user confirm account creation before authenticating"""
  136. if user:
  137. return None
  138. request = strategy.request
  139. backend_name = get_social_auth_backend_name(backend.name)
  140. if request.method == 'POST':
  141. try:
  142. request_data = json.loads(request.body)
  143. except (TypeError, ValueError):
  144. request_data = request.POST.copy()
  145. form = SocialAuthRegisterForm(
  146. request_data,
  147. request=request,
  148. agreements=Agreement.objects.get_agreements(),
  149. )
  150. if not form.is_valid():
  151. return JsonResponse(form.errors, status=400)
  152. email_verified = form.cleaned_data['email'] == details.get('email')
  153. activation_kwargs = {}
  154. if settings.account_activation == 'admin':
  155. activation_kwargs = {'requires_activation': UserModel.ACTIVATION_ADMIN}
  156. elif settings.account_activation == 'user' and not email_verified:
  157. activation_kwargs = {'requires_activation': UserModel.ACTIVATION_USER}
  158. try:
  159. new_user = UserModel.objects.create_user(
  160. form.cleaned_data['username'],
  161. form.cleaned_data['email'],
  162. create_audit_trail=True,
  163. joined_from_ip=request.user_ip,
  164. set_default_avatar=True,
  165. **activation_kwargs
  166. )
  167. except IntegrityError:
  168. return JsonResponse({'__all__': _("Please try resubmitting the form.")}, status=400)
  169. save_user_agreements(new_user, form)
  170. send_welcome_email(request, new_user)
  171. return {'user': new_user, 'is_new': True}
  172. request.frontend_context['SOCIAL_AUTH'] = {
  173. 'backend_name': backend_name,
  174. 'step': 'register',
  175. 'email': details.get('email'),
  176. 'username': kwargs.get('clean_username'),
  177. 'url': reverse('social:complete', kwargs={'backend': backend.name}),
  178. }
  179. return render(request, 'misago/socialauth.html', {
  180. 'backend_name': backend_name,
  181. })
  182. @partial
  183. def require_activation(strategy, details, backend, user=None, is_new=False, *args, **kwargs):
  184. if not user:
  185. # Social auth pipeline has entered corrupted state
  186. # Remove partial auth state and redirect user to beginning
  187. partial_token = strategy.session.get('partial_pipeline_token')
  188. if partial_token:
  189. strategy.clean_partial_pipeline(partial_token)
  190. return None
  191. if not user.requires_activation:
  192. return None
  193. request = strategy.request
  194. backend_name = get_social_auth_backend_name(backend.name)
  195. response_data = get_registration_result_json(user)
  196. response_data.update({'step': 'done', 'backend_name': backend_name})
  197. if request.method == 'POST':
  198. # we are carrying on from requestration request
  199. return JsonResponse(response_data)
  200. request.frontend_context['SOCIAL_AUTH'] = response_data
  201. request.frontend_context['SOCIAL_AUTH'].update({
  202. 'url': reverse('social:complete', kwargs={'backend': backend.name}),
  203. })
  204. return render(request, 'misago/socialauth.html', {
  205. 'backend_name': backend_name,
  206. })