user.py 16 KB

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