user.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483
  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('Rank', null=True, blank=True, on_delete=models.deletion.PROTECT)
  140. title = models.CharField(max_length=255, null=True, blank=True)
  141. requires_activation = models.PositiveIntegerField(default=ACTIVATION_NONE)
  142. is_staff = models.BooleanField(
  143. _('staff status'),
  144. default=False,
  145. help_text=_('Designates whether the user can log into admin sites.'),
  146. )
  147. roles = models.ManyToManyField('misago_acl.Role')
  148. acl_key = models.CharField(max_length=12, null=True, blank=True)
  149. is_active = models.BooleanField(
  150. _('active'),
  151. db_index=True,
  152. default=True,
  153. help_text=_(
  154. "Designates whether this user should be treated as active. "
  155. "Unselect this instead of deleting accounts."
  156. ),
  157. )
  158. is_active_staff_message = models.TextField(null=True, blank=True)
  159. avatar_tmp = models.ImageField(
  160. max_length=255,
  161. upload_to=avatars.store.upload_to,
  162. null=True,
  163. blank=True,
  164. )
  165. avatar_src = models.ImageField(
  166. max_length=255,
  167. upload_to=avatars.store.upload_to,
  168. null=True,
  169. blank=True,
  170. )
  171. avatar_crop = models.CharField(max_length=255, null=True, blank=True)
  172. avatars = JSONField(null=True, blank=True)
  173. is_avatar_locked = models.BooleanField(default=False)
  174. avatar_lock_user_message = models.TextField(null=True, blank=True)
  175. avatar_lock_staff_message = models.TextField(null=True, blank=True)
  176. signature = models.TextField(null=True, blank=True)
  177. signature_parsed = models.TextField(null=True, blank=True)
  178. signature_checksum = models.CharField(max_length=64, null=True, blank=True)
  179. is_signature_locked = models.BooleanField(default=False)
  180. signature_lock_user_message = models.TextField(null=True, blank=True)
  181. signature_lock_staff_message = models.TextField(null=True, blank=True)
  182. followers = models.PositiveIntegerField(default=0)
  183. following = models.PositiveIntegerField(default=0)
  184. follows = models.ManyToManyField(
  185. 'self',
  186. related_name='followed_by',
  187. symmetrical=False,
  188. )
  189. blocks = models.ManyToManyField(
  190. 'self',
  191. related_name='blocked_by',
  192. symmetrical=False,
  193. )
  194. limits_private_thread_invites_to = models.PositiveIntegerField(
  195. default=LIMIT_INVITES_TO_NONE,
  196. choices=LIMIT_INVITES_TO_CHOICES,
  197. )
  198. unread_private_threads = models.PositiveIntegerField(default=0)
  199. sync_unread_private_threads = models.BooleanField(default=False)
  200. subscribe_to_started_threads = models.PositiveIntegerField(
  201. default=SUBSCRIBE_NONE,
  202. choices=SUBSCRIBE_CHOICES,
  203. )
  204. subscribe_to_replied_threads = models.PositiveIntegerField(
  205. default=SUBSCRIBE_NONE,
  206. choices=SUBSCRIBE_CHOICES,
  207. )
  208. threads = models.PositiveIntegerField(default=0)
  209. posts = models.PositiveIntegerField(default=0, db_index=True)
  210. last_posted_on = models.DateTimeField(null=True, blank=True)
  211. USERNAME_FIELD = 'slug'
  212. REQUIRED_FIELDS = ['email']
  213. objects = UserManager()
  214. def clean(self):
  215. self.username = self.normalize_username(self.username)
  216. self.email = self.__class__.objects.normalize_email(self.email)
  217. def lock(self):
  218. """locks user in DB, shortcut for locking user model in views"""
  219. return User.objects.select_for_update().get(pk=self.pk)
  220. def delete(self, *args, **kwargs):
  221. if kwargs.pop('delete_content', False):
  222. self.delete_content()
  223. avatars.delete_avatar(self)
  224. return super(User, self).delete(*args, **kwargs)
  225. def delete_content(self):
  226. from misago.users.signals import delete_user_content
  227. delete_user_content.send(sender=self)
  228. @property
  229. def acl_cache(self):
  230. try:
  231. return self._acl_cache
  232. except AttributeError:
  233. self._acl_cache = get_user_acl(self)
  234. return self._acl_cache
  235. @acl_cache.setter
  236. def acl_cache(self, value):
  237. raise TypeError("acl_cache can't be assigned")
  238. @property
  239. def acl_(self):
  240. raise NotImplementedError('user.acl_ property was renamed to user.acl')
  241. @property
  242. def requires_activation_by_admin(self):
  243. return self.requires_activation == self.ACTIVATION_ADMIN
  244. @property
  245. def requires_activation_by_user(self):
  246. return self.requires_activation == self.ACTIVATION_USER
  247. @property
  248. def can_be_messaged_by_everyone(self):
  249. preference = self.limits_private_thread_invites_to
  250. return preference == self.LIMIT_INVITES_TO_NONE
  251. @property
  252. def can_be_messaged_by_followed(self):
  253. preference = self.limits_private_thread_invites_to
  254. return preference == self.LIMIT_INVITES_TO_FOLLOWED
  255. @property
  256. def can_be_messaged_by_nobody(self):
  257. preference = self.limits_private_thread_invites_to
  258. return preference == self.LIMIT_INVITES_TO_NOBODY
  259. @property
  260. def has_valid_signature(self):
  261. return is_user_signature_valid(self)
  262. def get_absolute_url(self):
  263. return reverse(
  264. 'misago:user', kwargs={
  265. 'slug': self.slug,
  266. 'pk': self.pk,
  267. }
  268. )
  269. def get_username(self):
  270. """dirty hack: return real username instead of normalized slug"""
  271. return self.username
  272. def get_full_name(self):
  273. return self.username
  274. def get_short_name(self):
  275. return self.username
  276. def set_username(self, new_username, changed_by=None):
  277. new_username = self.normalize_username(new_username)
  278. if new_username != self.username:
  279. old_username = self.username
  280. self.username = new_username
  281. self.slug = slugify(new_username)
  282. if self.pk:
  283. changed_by = changed_by or self
  284. self.record_name_change(changed_by, new_username, old_username)
  285. from misago.users.signals import username_changed
  286. username_changed.send(sender=self)
  287. def record_name_change(self, changed_by, new_username, old_username):
  288. self.namechanges.create(
  289. new_username=new_username,
  290. old_username=old_username,
  291. changed_by=changed_by,
  292. changed_by_username=changed_by.username,
  293. )
  294. def set_email(self, new_email):
  295. self.email = UserManager.normalize_email(new_email)
  296. self.email_hash = hash_email(new_email)
  297. def get_any_title(self):
  298. return self.title or self.rank.title or self.rank.name
  299. def get_roles(self):
  300. roles_pks = []
  301. roles_dict = {}
  302. for role in self.roles.all():
  303. roles_pks.append(role.pk)
  304. role.origin = self
  305. roles_dict[role.pk] = role
  306. if self.rank:
  307. for role in self.rank.roles.all():
  308. if role.pk not in roles_pks:
  309. role.origin = self.rank
  310. roles_pks.append(role.pk)
  311. roles_dict[role.pk] = role
  312. return [roles_dict[r] for r in sorted(roles_pks)]
  313. def update_acl_key(self):
  314. roles_pks = []
  315. for role in self.get_roles():
  316. if role.origin == 'self':
  317. roles_pks.append('u%s' % role.pk)
  318. else:
  319. roles_pks.append('%s:%s' % (self.rank.pk, role.pk))
  320. self.acl_key = md5(','.join(roles_pks).encode()).hexdigest()[:12]
  321. def email_user(self, subject, message, from_email=None, **kwargs):
  322. """sends an email to this user (for compat with Django)"""
  323. send_mail(subject, message, from_email, [self.email], **kwargs)
  324. def is_following(self, user):
  325. try:
  326. self.follows.get(pk=user.pk)
  327. return True
  328. except User.DoesNotExist:
  329. return False
  330. def is_blocking(self, user):
  331. try:
  332. self.blocks.get(pk=user.pk)
  333. return True
  334. except User.DoesNotExist:
  335. return False
  336. class Online(models.Model):
  337. user = models.OneToOneField(
  338. settings.AUTH_USER_MODEL,
  339. primary_key=True,
  340. related_name='online_tracker',
  341. )
  342. current_ip = models.GenericIPAddressField()
  343. last_click = models.DateTimeField(default=timezone.now)
  344. def save(self, *args, **kwargs):
  345. try:
  346. super(Online, self).save(*args, **kwargs)
  347. except IntegrityError:
  348. pass # first come is first serve in online tracker
  349. class UsernameChange(models.Model):
  350. user = models.ForeignKey(settings.AUTH_USER_MODEL, related_name='namechanges')
  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. try:
  372. return self._acl_cache
  373. except AttributeError:
  374. self._acl_cache = get_user_acl(self)
  375. return self._acl_cache
  376. @acl_cache.setter
  377. def acl_cache(self, value):
  378. raise TypeError("AnonymousUser instances can't be made ACL aware")
  379. def get_roles(self):
  380. try:
  381. return [Role.objects.get(special_role="anonymous")]
  382. except Role.DoesNotExist:
  383. raise RuntimeError("Anonymous user role not found.")
  384. def update_acl_key(self):
  385. raise TypeError("Can't update ACL key on anonymous users")