pipeline.py 8.1 KB

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