user.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452
  1. from hashlib import md5
  2. from django.contrib.auth.models import AnonymousUser as DjangoAnonymousUser
  3. from django.contrib.auth.models import UserManager as BaseUserManager
  4. from django.contrib.auth.models import AbstractBaseUser, PermissionsMixin
  5. from django.contrib.auth.password_validation import validate_password
  6. from django.contrib.postgres.fields import ArrayField, HStoreField, JSONField
  7. from django.core.mail import send_mail
  8. from django.db import IntegrityError, models, transaction
  9. from django.urls import reverse
  10. from django.utils import timezone
  11. from django.utils.translation import gettext_lazy as _
  12. from misago.acl.models import Role
  13. from misago.conf import settings
  14. from misago.core.pgutils import PgPartialIndex
  15. from misago.core.utils import slugify
  16. from misago.users import avatars
  17. from misago.users.audittrail import create_user_audit_trail
  18. from misago.users.signatures import is_user_signature_valid
  19. from misago.users.utils import hash_email
  20. from .online import Online
  21. from .rank import Rank
  22. class UserManager(BaseUserManager):
  23. def _create_user(self, username, email, password, **extra_fields):
  24. """
  25. Create and save a user with the given username, email, and password.
  26. """
  27. if not username:
  28. raise ValueError("User must have an username.")
  29. if not email:
  30. raise ValueError("User must have an email address.")
  31. user = self.model(**extra_fields)
  32. user.set_username(username)
  33. user.set_email(email)
  34. user.set_password(password)
  35. if not "rank" in extra_fields:
  36. user.rank = Rank.objects.get_default()
  37. now = timezone.now()
  38. user.last_login = now
  39. user.joined_on = now
  40. user.save(using=self._db)
  41. self._assert_user_has_authenticated_role(user)
  42. Online.objects.create(user=user, last_click=now)
  43. return user
  44. def _assert_user_has_authenticated_role(self, user):
  45. authenticated_role = Role.objects.get(special_role="authenticated")
  46. if authenticated_role not in user.roles.all():
  47. user.roles.add(authenticated_role)
  48. user.update_acl_key()
  49. user.save(update_fields=["acl_key"])
  50. def create_user(self, username, email=None, password=None, **extra_fields):
  51. extra_fields.setdefault("is_staff", False)
  52. extra_fields.setdefault("is_superuser", False)
  53. return self._create_user(username, email, password, **extra_fields)
  54. def create_superuser(self, username, email, password=None, **extra_fields):
  55. extra_fields.setdefault("is_staff", True)
  56. extra_fields.setdefault("is_superuser", True)
  57. if extra_fields.get("is_staff") is not True:
  58. raise ValueError("Superuser must have is_staff=True.")
  59. if extra_fields.get("is_superuser") is not True:
  60. raise ValueError("Superuser must have is_superuser=True.")
  61. try:
  62. if not extra_fields.get("rank"):
  63. extra_fields["rank"] = Rank.objects.get(name=_("Forum team"))
  64. except Rank.DoesNotExist:
  65. pass
  66. return self._create_user(username, email, password, **extra_fields)
  67. def get_by_username(self, username):
  68. return self.get(slug=slugify(username))
  69. def get_by_email(self, email):
  70. return self.get(email_hash=hash_email(email))
  71. def get_by_username_or_email(self, login):
  72. if "@" in login:
  73. return self.get(email_hash=hash_email(login))
  74. return self.get(slug=slugify(login))
  75. class User(AbstractBaseUser, PermissionsMixin):
  76. ACTIVATION_NONE = 0
  77. ACTIVATION_USER = 1
  78. ACTIVATION_ADMIN = 2
  79. SUBSCRIPTION_NONE = 0
  80. SUBSCRIPTION_NOTIFY = 1
  81. SUBSCRIPTION_ALL = 2
  82. SUBSCRIPTION_CHOICES = [
  83. (SUBSCRIPTION_NONE, _("No")),
  84. (SUBSCRIPTION_NOTIFY, _("Notify")),
  85. (SUBSCRIPTION_ALL, _("Notify with e-mail")),
  86. ]
  87. LIMIT_INVITES_TO_NONE = 0
  88. LIMIT_INVITES_TO_FOLLOWED = 1
  89. LIMIT_INVITES_TO_NOBODY = 2
  90. LIMIT_INVITES_TO_CHOICES = [
  91. (LIMIT_INVITES_TO_NONE, _("Everybody")),
  92. (LIMIT_INVITES_TO_FOLLOWED, _("Users I follow")),
  93. (LIMIT_INVITES_TO_NOBODY, _("Nobody")),
  94. ]
  95. # Note that "username" field is purely for shows.
  96. # When searching users by their names, always use lowercased string
  97. # and slug field instead that is normalized around DB engines
  98. # differences in case handling.
  99. username = models.CharField(max_length=30)
  100. slug = models.CharField(max_length=30, unique=True)
  101. # Misago stores user email in two fields:
  102. # "email" holds normalized email address
  103. # "email_hash" is lowercase hash of email address used to identify account
  104. # as well as enforcing on database level that no more than one user can be
  105. # using one email address
  106. email = models.EmailField(max_length=255, db_index=True)
  107. email_hash = models.CharField(max_length=32, unique=True)
  108. joined_on = models.DateTimeField(_("joined on"), default=timezone.now)
  109. joined_from_ip = models.GenericIPAddressField(null=True, blank=True)
  110. is_hiding_presence = models.BooleanField(default=False)
  111. rank = models.ForeignKey(
  112. "Rank", null=True, blank=True, on_delete=models.deletion.PROTECT
  113. )
  114. title = models.CharField(max_length=255, null=True, blank=True)
  115. requires_activation = models.PositiveIntegerField(default=ACTIVATION_NONE)
  116. is_staff = models.BooleanField(
  117. _("staff status"),
  118. default=False,
  119. help_text=_("Designates whether the user can log into admin sites."),
  120. )
  121. roles = models.ManyToManyField("misago_acl.Role")
  122. acl_key = models.CharField(max_length=12, null=True, blank=True)
  123. is_active = models.BooleanField(
  124. _("active"),
  125. db_index=True,
  126. default=True,
  127. help_text=_(
  128. "Designates whether this user should be treated as active. "
  129. "Unselect this instead of deleting accounts."
  130. ),
  131. )
  132. is_active_staff_message = models.TextField(null=True, blank=True)
  133. is_deleting_account = models.BooleanField(default=False)
  134. avatar_tmp = models.ImageField(
  135. max_length=255, upload_to=avatars.store.upload_to, null=True, blank=True
  136. )
  137. avatar_src = models.ImageField(
  138. max_length=255, upload_to=avatars.store.upload_to, null=True, blank=True
  139. )
  140. avatar_crop = models.CharField(max_length=255, null=True, blank=True)
  141. avatars = JSONField(null=True, blank=True)
  142. is_avatar_locked = models.BooleanField(default=False)
  143. avatar_lock_user_message = models.TextField(null=True, blank=True)
  144. avatar_lock_staff_message = models.TextField(null=True, blank=True)
  145. signature = models.TextField(null=True, blank=True)
  146. signature_parsed = models.TextField(null=True, blank=True)
  147. signature_checksum = models.CharField(max_length=64, null=True, blank=True)
  148. is_signature_locked = models.BooleanField(default=False)
  149. signature_lock_user_message = models.TextField(null=True, blank=True)
  150. signature_lock_staff_message = models.TextField(null=True, blank=True)
  151. followers = models.PositiveIntegerField(default=0)
  152. following = models.PositiveIntegerField(default=0)
  153. follows = models.ManyToManyField(
  154. "self", related_name="followed_by", symmetrical=False
  155. )
  156. blocks = models.ManyToManyField(
  157. "self", related_name="blocked_by", symmetrical=False
  158. )
  159. limits_private_thread_invites_to = models.PositiveIntegerField(
  160. default=LIMIT_INVITES_TO_NONE, choices=LIMIT_INVITES_TO_CHOICES
  161. )
  162. unread_private_threads = models.PositiveIntegerField(default=0)
  163. sync_unread_private_threads = models.BooleanField(default=False)
  164. subscribe_to_started_threads = models.PositiveIntegerField(
  165. default=SUBSCRIPTION_NONE, choices=SUBSCRIPTION_CHOICES
  166. )
  167. subscribe_to_replied_threads = models.PositiveIntegerField(
  168. default=SUBSCRIPTION_NONE, choices=SUBSCRIPTION_CHOICES
  169. )
  170. threads = models.PositiveIntegerField(default=0)
  171. posts = models.PositiveIntegerField(default=0, db_index=True)
  172. last_posted_on = models.DateTimeField(null=True, blank=True)
  173. profile_fields = HStoreField(default=dict)
  174. agreements = ArrayField(models.PositiveIntegerField(), default=list)
  175. USERNAME_FIELD = "slug"
  176. REQUIRED_FIELDS = ["email"]
  177. objects = UserManager()
  178. class Meta:
  179. indexes = [
  180. PgPartialIndex(fields=["is_staff"], where={"is_staff": True}),
  181. PgPartialIndex(
  182. fields=["requires_activation"], where={"requires_activation__gt": 0}
  183. ),
  184. PgPartialIndex(
  185. fields=["is_deleting_account"], where={"is_deleting_account": True}
  186. ),
  187. ]
  188. def clean(self):
  189. self.username = self.normalize_username(self.username)
  190. self.email = UserManager.normalize_email(self.email)
  191. def lock(self):
  192. """locks user in DB, shortcut for locking user model in views"""
  193. return User.objects.select_for_update().get(pk=self.pk)
  194. def delete(self, *args, **kwargs):
  195. if kwargs.pop("delete_content", False):
  196. self.delete_content()
  197. self.anonymize_data()
  198. avatars.delete_avatar(self)
  199. return super().delete(*args, **kwargs)
  200. def delete_content(self):
  201. from misago.users.signals import delete_user_content
  202. delete_user_content.send(sender=self)
  203. def mark_for_delete(self):
  204. self.is_active = False
  205. self.is_deleting_account = True
  206. self.save(update_fields=["is_active", "is_deleting_account"])
  207. def anonymize_data(self):
  208. """Replaces username with anonymized one, then send anonymization signal.
  209. Items associated with this user then anonymize their user-specific data
  210. like username or IP addresses.
  211. """
  212. self.username = settings.MISAGO_ANONYMOUS_USERNAME
  213. self.slug = slugify(self.username)
  214. from misago.users.signals import anonymize_user_data
  215. anonymize_user_data.send(sender=self)
  216. @property
  217. def requires_activation_by_admin(self):
  218. return self.requires_activation == self.ACTIVATION_ADMIN
  219. @property
  220. def requires_activation_by_user(self):
  221. return self.requires_activation == self.ACTIVATION_USER
  222. @property
  223. def can_be_messaged_by_everyone(self):
  224. preference = self.limits_private_thread_invites_to
  225. return preference == self.LIMIT_INVITES_TO_NONE
  226. @property
  227. def can_be_messaged_by_followed(self):
  228. preference = self.limits_private_thread_invites_to
  229. return preference == self.LIMIT_INVITES_TO_FOLLOWED
  230. @property
  231. def can_be_messaged_by_nobody(self):
  232. preference = self.limits_private_thread_invites_to
  233. return preference == self.LIMIT_INVITES_TO_NOBODY
  234. @property
  235. def has_valid_signature(self):
  236. return is_user_signature_valid(self)
  237. def get_absolute_url(self):
  238. return reverse("misago:user", kwargs={"slug": self.slug, "pk": self.pk})
  239. def get_username(self):
  240. """dirty hack: return real username instead of normalized slug"""
  241. return self.username
  242. def get_full_name(self):
  243. return self.username
  244. def get_short_name(self):
  245. return self.username
  246. def get_real_name(self):
  247. return self.profile_fields.get("real_name")
  248. def set_username(self, new_username, changed_by=None):
  249. new_username = self.normalize_username(new_username)
  250. if new_username != self.username:
  251. old_username = self.username
  252. self.username = new_username
  253. self.slug = slugify(new_username)
  254. if self.pk:
  255. changed_by = changed_by or self
  256. namechange = self.record_name_change(
  257. changed_by, new_username, old_username
  258. )
  259. from misago.users.signals import username_changed
  260. username_changed.send(sender=self)
  261. return namechange
  262. def record_name_change(self, changed_by, new_username, old_username):
  263. return self.namechanges.create(
  264. new_username=new_username,
  265. old_username=old_username,
  266. changed_by=changed_by,
  267. changed_by_username=changed_by.username,
  268. )
  269. def set_email(self, new_email):
  270. self.email = UserManager.normalize_email(new_email)
  271. self.email_hash = hash_email(new_email)
  272. def get_any_title(self):
  273. return self.title or self.rank.title or self.rank.name
  274. def get_roles(self):
  275. roles_pks = []
  276. roles_dict = {}
  277. for role in self.roles.all():
  278. roles_pks.append(role.pk)
  279. role.origin = self
  280. roles_dict[role.pk] = role
  281. if self.rank:
  282. for role in self.rank.roles.all():
  283. if role.pk not in roles_pks:
  284. role.origin = self.rank
  285. roles_pks.append(role.pk)
  286. roles_dict[role.pk] = role
  287. return [roles_dict[r] for r in sorted(roles_pks)]
  288. def update_acl_key(self):
  289. roles_pks = []
  290. for role in self.get_roles():
  291. if role.origin == "self":
  292. roles_pks.append("u%s" % role.pk)
  293. else:
  294. roles_pks.append("%s:%s" % (self.rank.pk, role.pk))
  295. self.acl_key = md5(",".join(roles_pks).encode()).hexdigest()[:12]
  296. def email_user(self, subject, message, from_email=None, **kwargs):
  297. """sends an email to this user (for compat with Django)"""
  298. send_mail(subject, message, from_email, [self.email], **kwargs)
  299. def is_following(self, user_or_id):
  300. try:
  301. user_id = user_or_id.id
  302. except AttributeError:
  303. user_id = user_or_id
  304. try:
  305. self.follows.get(id=user_id)
  306. return True
  307. except User.DoesNotExist:
  308. return False
  309. def is_blocking(self, user_or_id):
  310. try:
  311. user_id = user_or_id.id
  312. except AttributeError:
  313. user_id = user_or_id
  314. try:
  315. self.blocks.get(id=user_id)
  316. return True
  317. except User.DoesNotExist:
  318. return False
  319. class UsernameChange(models.Model):
  320. user = models.ForeignKey(
  321. settings.AUTH_USER_MODEL, related_name="namechanges", on_delete=models.CASCADE
  322. )
  323. changed_by = models.ForeignKey(
  324. settings.AUTH_USER_MODEL,
  325. null=True,
  326. blank=True,
  327. related_name="user_renames",
  328. on_delete=models.SET_NULL,
  329. )
  330. changed_by_username = models.CharField(max_length=30)
  331. changed_on = models.DateTimeField(default=timezone.now)
  332. new_username = models.CharField(max_length=255)
  333. old_username = models.CharField(max_length=255)
  334. class Meta:
  335. get_latest_by = "changed_on"
  336. def set_change_author(self, user):
  337. self.changed_by = user
  338. self.changed_by_username = user.username
  339. class AnonymousUser(DjangoAnonymousUser):
  340. acl_key = "anonymous"
  341. @property
  342. def acl_cache(self):
  343. raise Exception("AnonymousUser.acl_cache has been removed")
  344. @acl_cache.setter
  345. def acl_cache(self, value):
  346. raise TypeError("AnonymousUser instances can't be made ACL aware")
  347. def get_roles(self):
  348. try:
  349. return [Role.objects.get(special_role="anonymous")]
  350. except Role.DoesNotExist:
  351. raise RuntimeError("Anonymous user role not found.")
  352. def update_acl_key(self):
  353. raise TypeError("Can't update ACL key on anonymous users")