models.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471
  1. import hashlib
  2. import math
  3. from random import choice
  4. from path import path
  5. from django.conf import settings
  6. from django.contrib.auth.hashers import (
  7. check_password, make_password, is_password_usable, UNUSABLE_PASSWORD)
  8. from django.core.cache import cache, InvalidCacheBackendError
  9. from django.core.exceptions import ValidationError
  10. from django.core.mail import EmailMultiAlternatives
  11. from django.db import models
  12. from django.template import RequestContext
  13. from django.utils import timezone as tz_util
  14. from django.utils.translation import ugettext_lazy as _
  15. from misago.acl.builder import build_acl
  16. from misago.monitor.monitor import Monitor
  17. from misago.roles.models import Role
  18. from misago.settings.settings import Settings as DBSettings
  19. from misago.users.validators import validate_username, validate_password, validate_email
  20. from misago.utils import get_random_string, slugify
  21. class UserManager(models.Manager):
  22. """
  23. User Manager provides us with some additional methods for users
  24. """
  25. def get_blank_user(self):
  26. blank_user = User(
  27. join_date=tz_util.now(),
  28. join_ip='127.0.0.1'
  29. )
  30. return blank_user
  31. def resync_monitor(self, monitor):
  32. monitor['users'] = self.count()
  33. monitor['users_inactive'] = self.filter(activation__gt=0).count()
  34. last_user = self.latest('id')
  35. monitor['last_user'] = last_user.pk
  36. monitor['last_user_name'] = last_user.username
  37. monitor['last_user_slug'] = last_user.username_slug
  38. def create_user(self, username, email, password, timezone=False, ip='127.0.0.1', no_roles=False, activation=0, request=False):
  39. token = ''
  40. if activation > 0:
  41. token = get_random_string(12)
  42. try:
  43. db_settings = request.settings
  44. except AttributeError:
  45. db_settings = DBSettings()
  46. if timezone == False:
  47. timezone = db_settings['default_timezone']
  48. # Get first rank
  49. try:
  50. from misago.ranks.models import Rank
  51. default_rank = Rank.objects.filter(special=0).order_by('order')[0]
  52. except Rank.DoesNotExist:
  53. default_rank = None
  54. # Store user in database
  55. new_user = User(
  56. last_sync=tz_util.now(),
  57. join_date=tz_util.now(),
  58. join_ip=ip,
  59. activation=activation,
  60. token=token,
  61. timezone=timezone,
  62. rank=default_rank,
  63. )
  64. new_user.set_username(username)
  65. new_user.set_email(email)
  66. new_user.set_password(password)
  67. new_user.full_clean()
  68. new_user.default_avatar(db_settings)
  69. new_user.save(force_insert=True)
  70. # Set user roles?
  71. if not no_roles:
  72. from misago.roles.models import Role
  73. new_user.roles.add(Role.objects.get(token='registered'))
  74. new_user.make_acl_key()
  75. new_user.save(force_update=True)
  76. # Load monitor
  77. try:
  78. monitor = request.monitor
  79. except AttributeError:
  80. monitor = Monitor()
  81. # Update forum stats
  82. if activation == 0:
  83. monitor['users'] = int(monitor['users']) + 1
  84. monitor['last_user'] = new_user.pk
  85. monitor['last_user_name'] = new_user.username
  86. monitor['last_user_slug'] = new_user.username_slug
  87. else:
  88. monitor['users_inactive'] = int(monitor['users_inactive']) + 1
  89. # Return new user
  90. return new_user
  91. def get_by_email(self, email):
  92. return self.get(email_hash=hashlib.md5(email).hexdigest())
  93. def filter_stats(self, start, end):
  94. return self.filter(join_date__gte=start).filter(join_date__lte=end)
  95. class User(models.Model):
  96. """
  97. Misago User model
  98. """
  99. username = models.CharField(max_length=255,validators=[validate_username])
  100. username_slug = models.SlugField(max_length=255,unique=True,
  101. error_messages={'unique': _("This user name is already in use by another user.")})
  102. email = models.EmailField(max_length=255,validators=[validate_email])
  103. email_hash = models.CharField(max_length=32,unique=True,
  104. error_messages={'unique': _("This email address is already in use by another user.")})
  105. password = models.CharField(max_length=255)
  106. password_date = models.DateTimeField()
  107. avatar_type = models.CharField(max_length=10,null=True,blank=True)
  108. avatar_image = models.CharField(max_length=255,null=True,blank=True)
  109. signature = models.TextField(null=True,blank=True)
  110. signature_preparsed = models.TextField(null=True,blank=True)
  111. join_date = models.DateTimeField()
  112. join_ip = models.GenericIPAddressField()
  113. join_agent = models.TextField(null=True,blank=True)
  114. last_date = models.DateTimeField(null=True,blank=True)
  115. last_ip = models.GenericIPAddressField(null=True,blank=True)
  116. last_agent = models.TextField(null=True,blank=True)
  117. hide_activity = models.PositiveIntegerField(default=0)
  118. alert_ats = models.PositiveIntegerField(default=0)
  119. allow_pms = models.PositiveIntegerField(default=0)
  120. receive_newsletters = models.BooleanField(default=True)
  121. topics = models.PositiveIntegerField(default=0)
  122. posts = models.PositiveIntegerField(default=0)
  123. votes = models.PositiveIntegerField(default=0)
  124. karma_given_p = models.PositiveIntegerField(default=0)
  125. karma_given_n = models.PositiveIntegerField(default=0)
  126. karma_p = models.PositiveIntegerField(default=0)
  127. karma_n = models.PositiveIntegerField(default=0)
  128. following = models.PositiveIntegerField(default=0)
  129. followers = models.PositiveIntegerField(default=0)
  130. score = models.IntegerField(default=0,db_index=True)
  131. rank = models.ForeignKey('ranks.Rank',null=True,blank=True,db_index=True,on_delete=models.SET_NULL)
  132. last_sync = models.DateTimeField(null=True,blank=True)
  133. follows = models.ManyToManyField('self',related_name='follows_set',symmetrical=False)
  134. ignores = models.ManyToManyField('self',related_name='ignores_set',symmetrical=False)
  135. title = models.CharField(max_length=255,null=True,blank=True)
  136. last_post = models.DateTimeField(null=True,blank=True)
  137. last_search = models.DateTimeField(null=True,blank=True)
  138. alerts = models.PositiveIntegerField(default=0)
  139. alerts_new = models.PositiveIntegerField(default=0)
  140. activation = models.IntegerField(default=0)
  141. token = models.CharField(max_length=12,null=True,blank=True)
  142. avatar_ban = models.BooleanField(default=False)
  143. avatar_ban_reason_user = models.TextField(null=True,blank=True)
  144. avatar_ban_reason_admin = models.TextField(null=True,blank=True)
  145. signature_ban = models.BooleanField(default=False)
  146. signature_ban_reason_user = models.TextField(null=True,blank=True)
  147. signature_ban_reason_admin = models.TextField(null=True,blank=True)
  148. timezone = models.CharField(max_length=255,default='utc')
  149. roles = models.ManyToManyField('roles.Role')
  150. is_team = models.BooleanField(default=False,db_index=True)
  151. acl_key = models.CharField(max_length=12,null=True,blank=True)
  152. objects = UserManager()
  153. ACTIVATION_NONE = 0
  154. ACTIVATION_USER = 1
  155. ACTIVATION_ADMIN = 2
  156. ACTIVATION_CREDENTIALS = 3
  157. statistics_name = _('Users Registrations')
  158. def is_god(self):
  159. try:
  160. return self.is_god_cache
  161. except AttributeError:
  162. for user in settings.ADMINS:
  163. if user[1].lower() == self.email:
  164. self.is_god_cache = True
  165. return True
  166. self.is_god_cache = False
  167. return False
  168. def is_anonymous(self):
  169. return False
  170. def is_authenticated(self):
  171. return True
  172. def is_crawler(self):
  173. return False
  174. def is_protected(self):
  175. for role in self.roles.all():
  176. if role.protected:
  177. return True
  178. return False
  179. def lock_avatar(self):
  180. # Kill existing avatar and lock our ability to change it
  181. self.delete_avatar()
  182. self.avatar_ban = True
  183. # Pick new one from _locked gallery
  184. galleries = path(settings.STATICFILES_DIRS[0]).joinpath('avatars').joinpath('_locked')
  185. avatars_list = galleries.files('*.gif')
  186. avatars_list += galleries.files('*.jpg')
  187. avatars_list += galleries.files('*.jpeg')
  188. avatars_list += galleries.files('*.png')
  189. self.avatar_type = 'gallery'
  190. self.avatar_image = '/'.join(path(choice(avatars_list)).splitall()[-2:])
  191. def default_avatar(self, db_settings):
  192. if db_settings['default_avatar'] == 'gallery':
  193. try:
  194. avatars_list = []
  195. try:
  196. # First try, _default path
  197. galleries = path(settings.STATICFILES_DIRS[0]).joinpath('avatars').joinpath('_default')
  198. avatars_list += galleries.files('*.gif')
  199. avatars_list += galleries.files('*.jpg')
  200. avatars_list += galleries.files('*.jpeg')
  201. avatars_list += galleries.files('*.png')
  202. except Exception as e:
  203. pass
  204. # Second try, all paths
  205. if not avatars_list:
  206. avatars_list = []
  207. for directory in path(settings.STATICFILES_DIRS[0]).joinpath('avatars').dirs():
  208. if not directory[-7:] == '_locked':
  209. avatars_list += directory.files('*.gif')
  210. avatars_list += directory.files('*.jpg')
  211. avatars_list += directory.files('*.jpeg')
  212. avatars_list += directory.files('*.png')
  213. if avatars_list:
  214. # Pick random avatar from list
  215. self.avatar_type = 'gallery'
  216. self.avatar_image = '/'.join(path(choice(avatars_list)).splitall()[-2:])
  217. return True
  218. except Exception as e:
  219. pass
  220. self.avatar_type = 'gravatar'
  221. self.avatar_image = None
  222. return True
  223. def delete_avatar(self):
  224. if self.avatar_type == 'upload':
  225. # DELETE OUR AVATAR!!!
  226. pass
  227. def delete_content(self):
  228. if self.pk:
  229. for model_obj in models.get_models():
  230. try:
  231. model_obj.objects.delete_user_content(self)
  232. except AttributeError:
  233. pass
  234. def delete(self, *args, **kwargs):
  235. self.delete_avatar()
  236. super(User, self).delete(*args, **kwargs)
  237. def set_username(self, username):
  238. self.username = username.strip()
  239. self.username_slug = slugify(username)
  240. if self.pk:
  241. for model_obj in models.get_models():
  242. try:
  243. model_obj.objects.update_username(self)
  244. except AttributeError:
  245. pass
  246. def set_signature(self, signature):
  247. self.signature = signature.strip()
  248. self.signature_preparsed = ''
  249. if self.signature:
  250. import markdown
  251. self.signature_preparsed = markdown.markdown(signature, safe_mode='escape', output_format=settings.OUTPUT_FORMAT)
  252. def is_username_valid(self, e):
  253. try:
  254. raise ValidationError(e.message_dict['username'])
  255. except KeyError:
  256. pass
  257. try:
  258. raise ValidationError(e.message_dict['username_slug'])
  259. except KeyError:
  260. pass
  261. def is_email_valid(self, e):
  262. try:
  263. raise ValidationError(e.message_dict['email'])
  264. except KeyError:
  265. pass
  266. try:
  267. raise ValidationError(e.message_dict['email_hash'])
  268. except KeyError:
  269. pass
  270. def is_password_valid(self, e):
  271. try:
  272. raise ValidationError(e.message_dict['password'])
  273. except KeyError:
  274. pass
  275. def set_email(self, email):
  276. self.email = email.strip().lower()
  277. self.email_hash = hashlib.md5(self.email).hexdigest()
  278. def set_password(self, raw_password):
  279. self.password_date = tz_util.now()
  280. self.password = make_password(raw_password.strip())
  281. def set_last_visit(self, ip, agent, hidden=False):
  282. self.last_date = tz_util.now()
  283. self.last_ip = ip
  284. self.last_agent = agent
  285. self.last_hide = hidden
  286. def check_password(self, raw_password, mobile=False):
  287. """
  288. Returns a boolean of whether the raw_password was correct. Handles
  289. hashing formats behind the scenes.
  290. """
  291. def setter(raw_password):
  292. self.set_password(raw_password)
  293. self.save()
  294. # Is standard password allright?
  295. if check_password(raw_password, self.password, setter):
  296. return True
  297. # Check mobile password?
  298. if mobile:
  299. raw_password = raw_password[:1].lower() + raw_password[1:]
  300. else:
  301. password_reversed = u''
  302. for c in raw_password:
  303. r = c.upper()
  304. if r == c:
  305. r = c.lower()
  306. password_reversed += r
  307. raw_password = password_reversed
  308. return check_password(raw_password, self.password, setter)
  309. def get_roles(self):
  310. return self.roles.all()
  311. def make_acl_key(self):
  312. if self.acl_key:
  313. return self.acl_key
  314. roles_ids = []
  315. for role in self.roles.all():
  316. roles_ids.append(str(role.pk))
  317. self.acl_key = 'acl_%s' % hashlib.md5('_'.join(roles_ids)).hexdigest()[0:8]
  318. return self.acl_key
  319. def get_acl(self, request):
  320. try:
  321. acl = cache.get(self.acl_key)
  322. if acl.version != request.monitor.acl_version:
  323. raise InvalidCacheBackendError()
  324. except AttributeError, InvalidCacheBackendError:
  325. # build acl cache
  326. acl = build_acl(request, self.get_roles())
  327. cache.set(self.acl_key, acl, 2592000)
  328. return acl
  329. def get_avatar(self, size='normal'):
  330. # Get uploaded avatar
  331. if self.avatar_type == 'upload':
  332. return settings.MEDIA_URL + 'avatars/' + self.avatar_image
  333. # Get gallery avatar
  334. if self.avatar_type == 'gallery':
  335. return settings.STATIC_URL + 'avatars/' + self.avatar_image
  336. # No avatar found, get gravatar
  337. if size == 'big':
  338. size = 150;
  339. elif size == 'small':
  340. size = 64;
  341. elif size == 'tiny':
  342. size = 46;
  343. else:
  344. size = 100
  345. return 'http://www.gravatar.com/avatar/%s?s=%s' % (hashlib.md5(self.email).hexdigest(), size)
  346. def get_title(self):
  347. if self.title:
  348. return self.title
  349. if self.rank:
  350. return self.rank.title
  351. return None
  352. def get_style(self):
  353. if self.rank:
  354. return self.rank.style
  355. return ''
  356. def email_user(self, request, template, subject, context={}):
  357. templates = request.theme.get_email_templates(template)
  358. context = RequestContext(request, context)
  359. context['author'] = context['user']
  360. context['user'] = self
  361. # Set message recipient
  362. if settings.DEBUG and settings.CATCH_ALL_EMAIL_ADDRESS:
  363. recipient = settings.CATCH_ALL_EMAIL_ADDRESS
  364. else:
  365. recipient = self.email
  366. # Build and send message
  367. email = EmailMultiAlternatives(subject, templates[0].render(context), settings.EMAIL_HOST_USER, [recipient])
  368. email.attach_alternative(templates[1].render(context), "text/html")
  369. email.send()
  370. def get_activation(self):
  371. activations = ['none', 'user', 'admin', 'credentials']
  372. return activations[self.activation]
  373. def get_date(self):
  374. return self.join_date
  375. def sync_user(self):
  376. print 'SYNCING USER!'
  377. class Guest(object):
  378. """
  379. Misago Guest dummy
  380. """
  381. def is_anonymous(self):
  382. return True
  383. def is_authenticated(self):
  384. return False
  385. def is_crawler(self):
  386. return False
  387. def get_roles(self):
  388. return Role.objects.filter(token='guest')
  389. def make_acl_key(self):
  390. return 'acl_%s' % hashlib.md5(str(Role.objects.get(token='guest').pk)).hexdigest()[0:8]
  391. class Crawler(Guest):
  392. """
  393. Misago Crawler dummy
  394. """
  395. def __init__(self, username):
  396. self.username = username
  397. def is_anonymous(self):
  398. return True
  399. def is_authenticated(self):
  400. return False
  401. def is_crawler(self):
  402. return True