user.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477
  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',
  113. null=True,
  114. blank=True,
  115. on_delete=models.deletion.PROTECT,
  116. )
  117. title = models.CharField(max_length=255, null=True, blank=True)
  118. requires_activation = models.PositiveIntegerField(default=ACTIVATION_NONE)
  119. is_staff = models.BooleanField(
  120. _('staff status'),
  121. default=False,
  122. help_text=_('Designates whether the user can log into admin sites.'),
  123. )
  124. roles = models.ManyToManyField('misago_acl.Role')
  125. acl_key = models.CharField(max_length=12, null=True, blank=True)
  126. is_active = models.BooleanField(
  127. _('active'),
  128. db_index=True,
  129. default=True,
  130. help_text=_(
  131. "Designates whether this user should be treated as active. "
  132. "Unselect this instead of deleting accounts."
  133. ),
  134. )
  135. is_active_staff_message = models.TextField(null=True, blank=True)
  136. is_deleting_account = models.BooleanField(default=False)
  137. avatar_tmp = models.ImageField(
  138. max_length=255,
  139. upload_to=avatars.store.upload_to,
  140. null=True,
  141. blank=True,
  142. )
  143. avatar_src = models.ImageField(
  144. max_length=255,
  145. upload_to=avatars.store.upload_to,
  146. null=True,
  147. blank=True,
  148. )
  149. avatar_crop = models.CharField(max_length=255, null=True, blank=True)
  150. avatars = JSONField(null=True, blank=True)
  151. is_avatar_locked = models.BooleanField(default=False)
  152. avatar_lock_user_message = models.TextField(null=True, blank=True)
  153. avatar_lock_staff_message = models.TextField(null=True, blank=True)
  154. signature = models.TextField(null=True, blank=True)
  155. signature_parsed = models.TextField(null=True, blank=True)
  156. signature_checksum = models.CharField(max_length=64, null=True, blank=True)
  157. is_signature_locked = models.BooleanField(default=False)
  158. signature_lock_user_message = models.TextField(null=True, blank=True)
  159. signature_lock_staff_message = models.TextField(null=True, blank=True)
  160. followers = models.PositiveIntegerField(default=0)
  161. following = models.PositiveIntegerField(default=0)
  162. follows = models.ManyToManyField(
  163. 'self',
  164. related_name='followed_by',
  165. symmetrical=False,
  166. )
  167. blocks = models.ManyToManyField(
  168. 'self',
  169. related_name='blocked_by',
  170. symmetrical=False,
  171. )
  172. limits_private_thread_invites_to = models.PositiveIntegerField(
  173. default=LIMIT_INVITES_TO_NONE,
  174. choices=LIMIT_INVITES_TO_CHOICES,
  175. )
  176. unread_private_threads = models.PositiveIntegerField(default=0)
  177. sync_unread_private_threads = models.BooleanField(default=False)
  178. subscribe_to_started_threads = models.PositiveIntegerField(
  179. default=SUBSCRIPTION_NONE,
  180. choices=SUBSCRIPTION_CHOICES,
  181. )
  182. subscribe_to_replied_threads = models.PositiveIntegerField(
  183. default=SUBSCRIPTION_NONE,
  184. choices=SUBSCRIPTION_CHOICES,
  185. )
  186. threads = models.PositiveIntegerField(default=0)
  187. posts = models.PositiveIntegerField(default=0, db_index=True)
  188. last_posted_on = models.DateTimeField(null=True, blank=True)
  189. profile_fields = HStoreField(default=dict)
  190. agreements = ArrayField(models.PositiveIntegerField(), default=list)
  191. USERNAME_FIELD = 'slug'
  192. REQUIRED_FIELDS = ['email']
  193. objects = UserManager()
  194. class Meta:
  195. indexes = [
  196. PgPartialIndex(
  197. fields=['is_staff'],
  198. where={'is_staff': True},
  199. ),
  200. PgPartialIndex(
  201. fields=['requires_activation'],
  202. where={'requires_activation__gt': 0},
  203. ),
  204. PgPartialIndex(
  205. fields=['is_deleting_account'],
  206. where={'is_deleting_account': True},
  207. ),
  208. ]
  209. def clean(self):
  210. self.username = self.normalize_username(self.username)
  211. self.email = UserManager.normalize_email(self.email)
  212. def lock(self):
  213. """locks user in DB, shortcut for locking user model in views"""
  214. return User.objects.select_for_update().get(pk=self.pk)
  215. def delete(self, *args, **kwargs):
  216. if kwargs.pop('delete_content', False):
  217. self.delete_content()
  218. self.anonymize_data()
  219. avatars.delete_avatar(self)
  220. return super().delete(*args, **kwargs)
  221. def delete_content(self):
  222. from misago.users.signals import delete_user_content
  223. delete_user_content.send(sender=self)
  224. def mark_for_delete(self):
  225. self.is_active = False
  226. self.is_deleting_account = True
  227. self.save(update_fields=['is_active', 'is_deleting_account'])
  228. def anonymize_data(self):
  229. """Replaces username with anonymized one, then send anonymization signal.
  230. Items associated with this user then anonymize their user-specific data
  231. like username or IP addresses.
  232. """
  233. self.username = settings.MISAGO_ANONYMOUS_USERNAME
  234. self.slug = slugify(self.username)
  235. from misago.users.signals import anonymize_user_data
  236. anonymize_user_data.send(sender=self)
  237. @property
  238. def requires_activation_by_admin(self):
  239. return self.requires_activation == self.ACTIVATION_ADMIN
  240. @property
  241. def requires_activation_by_user(self):
  242. return self.requires_activation == self.ACTIVATION_USER
  243. @property
  244. def can_be_messaged_by_everyone(self):
  245. preference = self.limits_private_thread_invites_to
  246. return preference == self.LIMIT_INVITES_TO_NONE
  247. @property
  248. def can_be_messaged_by_followed(self):
  249. preference = self.limits_private_thread_invites_to
  250. return preference == self.LIMIT_INVITES_TO_FOLLOWED
  251. @property
  252. def can_be_messaged_by_nobody(self):
  253. preference = self.limits_private_thread_invites_to
  254. return preference == self.LIMIT_INVITES_TO_NOBODY
  255. @property
  256. def has_valid_signature(self):
  257. return is_user_signature_valid(self)
  258. def get_absolute_url(self):
  259. return reverse(
  260. 'misago:user', kwargs={
  261. 'slug': self.slug,
  262. 'pk': self.pk,
  263. }
  264. )
  265. def get_username(self):
  266. """dirty hack: return real username instead of normalized slug"""
  267. return self.username
  268. def get_full_name(self):
  269. return self.username
  270. def get_short_name(self):
  271. return self.username
  272. def get_real_name(self):
  273. return self.profile_fields.get('real_name')
  274. def set_username(self, new_username, changed_by=None):
  275. new_username = self.normalize_username(new_username)
  276. if new_username != self.username:
  277. old_username = self.username
  278. self.username = new_username
  279. self.slug = slugify(new_username)
  280. if self.pk:
  281. changed_by = changed_by or self
  282. namechange = self.record_name_change(
  283. changed_by, new_username, old_username
  284. )
  285. from misago.users.signals import username_changed
  286. username_changed.send(sender=self)
  287. return namechange
  288. def record_name_change(self, changed_by, new_username, old_username):
  289. return self.namechanges.create(
  290. new_username=new_username,
  291. old_username=old_username,
  292. changed_by=changed_by,
  293. changed_by_username=changed_by.username,
  294. )
  295. def set_email(self, new_email):
  296. self.email = UserManager.normalize_email(new_email)
  297. self.email_hash = hash_email(new_email)
  298. def get_any_title(self):
  299. return self.title or self.rank.title or self.rank.name
  300. def get_roles(self):
  301. roles_pks = []
  302. roles_dict = {}
  303. for role in self.roles.all():
  304. roles_pks.append(role.pk)
  305. role.origin = self
  306. roles_dict[role.pk] = role
  307. if self.rank:
  308. for role in self.rank.roles.all():
  309. if role.pk not in roles_pks:
  310. role.origin = self.rank
  311. roles_pks.append(role.pk)
  312. roles_dict[role.pk] = role
  313. return [roles_dict[r] for r in sorted(roles_pks)]
  314. def update_acl_key(self):
  315. roles_pks = []
  316. for role in self.get_roles():
  317. if role.origin == 'self':
  318. roles_pks.append('u%s' % role.pk)
  319. else:
  320. roles_pks.append('%s:%s' % (self.rank.pk, role.pk))
  321. self.acl_key = md5(','.join(roles_pks).encode()).hexdigest()[:12]
  322. def email_user(self, subject, message, from_email=None, **kwargs):
  323. """sends an email to this user (for compat with Django)"""
  324. send_mail(subject, message, from_email, [self.email], **kwargs)
  325. def is_following(self, user_or_id):
  326. try:
  327. user_id = user_or_id.id
  328. except AttributeError:
  329. user_id = user_or_id
  330. try:
  331. self.follows.get(id=user_id)
  332. return True
  333. except User.DoesNotExist:
  334. return False
  335. def is_blocking(self, user_or_id):
  336. try:
  337. user_id = user_or_id.id
  338. except AttributeError:
  339. user_id = user_or_id
  340. try:
  341. self.blocks.get(id=user_id)
  342. return True
  343. except User.DoesNotExist:
  344. return False
  345. class UsernameChange(models.Model):
  346. user = models.ForeignKey(
  347. settings.AUTH_USER_MODEL,
  348. related_name='namechanges',
  349. on_delete=models.CASCADE,
  350. )
  351. changed_by = models.ForeignKey(
  352. settings.AUTH_USER_MODEL,
  353. null=True,
  354. blank=True,
  355. related_name='user_renames',
  356. on_delete=models.SET_NULL,
  357. )
  358. changed_by_username = models.CharField(max_length=30)
  359. changed_on = models.DateTimeField(default=timezone.now)
  360. new_username = models.CharField(max_length=255)
  361. old_username = models.CharField(max_length=255)
  362. class Meta:
  363. get_latest_by = "changed_on"
  364. def set_change_author(self, user):
  365. self.changed_by = user
  366. self.changed_by_username = user.username
  367. class AnonymousUser(DjangoAnonymousUser):
  368. acl_key = 'anonymous'
  369. @property
  370. def acl_cache(self):
  371. raise Exception("AnonymousUser.acl_cache has been removed")
  372. @acl_cache.setter
  373. def acl_cache(self, value):
  374. raise TypeError("AnonymousUser instances can't be made ACL aware")
  375. def get_roles(self):
  376. try:
  377. return [Role.objects.get(special_role="anonymous")]
  378. except Role.DoesNotExist:
  379. raise RuntimeError("Anonymous user role not found.")
  380. def update_acl_key(self):
  381. raise TypeError("Can't update ACL key on anonymous users")